You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

8.1 KiB

开篇词 | 如何学习谷歌高性能 JavaScript 引擎V8

你好,我是李兵,《[浏览器工作原理与实践](https://time.geekbang.org/column/intro/216?utm_term=zeusQYFJN&utm_source=app&utm_medium=geektime&utm_campaign=216-end&utm_content=v8zhuanlankaipianci0316)》专栏的作者。在浏览器专栏中我们对浏览器的工作原理进行了详细的介绍其中也提到了V8是如何执行JavaScript代码的。很多朋友对这部分的学习意犹未尽因此我又回到了极客时间与你深入聊聊V8。

什么是V8

V8是JavaScript虚拟机的一种。我们可以简单地把JavaScript虚拟机理解成是一个翻译程序将人类能够理解的编程语言JavaScript,翻译成机器能够理解的机器语言。如下图所示:

上图中中间的“黑盒”就是JavaScript引擎V8。目前市面上有很多种JavaScript引擎诸如SpiderMonkey、V8、JavaScriptCore等。而由谷歌开发的开源项目V8是当下使用最广泛的JavaScript虚拟机全球有超过25亿台安卓设备而这些设备中都使用了Chrome浏览器所以我们写的JavaScript应用大都跑在V8上。

V8之所以拥有如此庞大的生态圈也和它许多革命性的设计是分不开的。

在V8出现之前所有的JavaScript虚拟机所采用的都是解释执行的方式这是JavaScript执行速度过慢的一个主要原因。而V8率先引入了即时编译JIT的双轮驱动的设计这是一种权衡策略混合编译执行和解释执行这两种手段给JavaScript的执行速度带来了极大的提升。

V8出现之后各大厂商也都在自己的JavaScript虚拟机中引入了JIT机制所以你会看到目前市面上JavaScript虚拟机都有着类似的架构。另外V8也是早于其他虚拟机引入了惰性编译、内联缓存、隐藏类等机制进一步优化了JavaScript代码的编译执行效率。

可以说V8的出现将JavaScript虚拟机技术推向了一个全新的高度。

即便V8具有诸多优点但我相信对于大部分同学来说V8虚拟机还只是一个黑盒我们将一段代码丢给这个黑盒它便会返回结果并没有深入了解过它的工作原理。

如果只是单纯使用JavaScript和调用 Web API并不了解虚拟机内部是怎样工作的在项目中遇到的很多问题很可能找不到解决的途径。比如有时项目的占用内存过高或者页面响应速度过慢又或者使用Node.js的时候导致任务被阻塞等问题都与V8的基本运行机制有关。如果你熟悉V8的工作机制就会有系统性的思路来解决这些问题。

另外V8的主要功能就是结合JavaScript语言的特性和本质来编译执行它。通过深入地学习V8你对JavaScript语言本质和设计思想会有很直观的感受。这些设计思想像是更加高级的工具你掌握了它就可以提升你的语言使用和架构设计水平。

如何学习V8

那么我们应该如何来学习V8呢

刚刚我们也说过V8的主要职责是用来执行JavaScript代码的那我们首先需要先了解JavaScript这门语言的基本特性和设计思想。

JavaScript借鉴了很多语言的特性比如C语言的基本语法、Java的类型系统和内存管理、Scheme的函数作为一等公民还有Self基于原型prototype的继承机制。毫无疑问JavaScript是一门非常优秀的语言特别是“原型继承机制”和“函数是一等公民”这两个设计。

不过JavaScript也是一门处处是坑的语言由于历史原因很多错误的或者不合理的设计都被延续至今比如使用new加构造函数来创建对象这种方式的背后隐藏了太多的细节非常容易增加代码出错概率而且也大大增加了新手的学习成本再比如初期的JavaScript没有块级作用域机制使得JavaScript需要采取变量提升的策略而变量提升又是非常反人性的设计。

V8是JavaScript的实现在学习V8工作原理时我们就要格外关注JavaScript这些独特的设计思想和特性背后的实现。比如为了实现函数是一等公民的特性JavaScript采取了基于对象的策略再比如为了实现原型继承V8为每个对象引入了__proto__属性。

深入分析过JavaScript语言之后我们就可以学习V8执行JavaScript代码的完整流程了。我们把这套流程称之为V8的编译流水线其完整流程如下图所示

编译流水线本身并不复杂但是其中涉及到了很多技术诸如JIT、延迟解析、隐藏类、内联缓存等等。这些技术决定着一段JavaScript代码能否正常执行以及代码的执行效率。

比如V8中使用的隐藏类Hide Class这是将JavaScript中动态类型转换为静态类型的一种技术可以消除动态类型的语言执行速度过慢的问题如果你熟悉V8的工作机制在你编写JavaScript时就能充分利用好隐藏类这种强大的优化特性写出更加高效的代码。

再比如V8实现了JavaScript代码的惰性解析目的是为了加速代码的启动速度通过对惰性解析机制的学习你可以优化你的代码更加适应这个机制从而提高程序性能。

要想充分了解V8是怎么工作的除了要分析编译流水线我们还需要了解另外两个非常重要的特性那就是事件循环系统垃圾回收机制。

事件循环系统和JavaScript中的难点——异步编程特性紧密相关。我们知道JavaScript是单线程的JavaScript代码都是在一个线程上执行如果同一时间发送了多个JavaScript执行的请求就需要排队也就是进行异步编程。

V8的事件循环系统会调度这些排队任务保证JavaScript代码被V8有序地执行。因此也可以说事件循环系统就是V8的心脏它驱动了V8的持续工作。

另外JavaScript是一种自动垃圾回收的语言V8在执行垃圾回收时会占用主线程的资源如果我们编写的程序频繁触发垃圾回收那么无疑会阻塞主线程这也是我们经常会遇到的一个问题。你需要知道V8是如何分配内存数据的以及这些数据是如何被回收的打通整个链路建立完整的系统当下次遇到内存问题时就知道如何去排查了。

以上就是系统学习V8的路径。在我们这一季的课程中也会按照这样的思路来设计课程来帮助你学习到V8的完整的知识体系。

  • 首先我们会从JavaScript的设计思想讲起讨论它背后的一些主要特性以及V8是怎么实现这些特性的。
  • 然后我们再来分析V8的编译流水线在课程中间我们还会穿插介绍一些内存分配相关的内容因为函数调用、变量声明、参数传递或者函数返回数值都涉及到了内存分配。
  • 最后,我们会介绍事件循环系统和垃圾回收系统的工作机制。

虽然本课程的篇幅不多,但是也具有一定的深度和广度。不过你并不需要担心内容太难,我会尽量将每节内容做到深入浅出,有什么问题你可以在留言区提问,我看到后都会第一时间来解答。

另外在每个模块结束后我会做一次热点问题的答疑尽量帮你扫清学习V8的障碍。你需要做的是持之以恒地学习、反思与实践。

加油从今天起就让我们一起开始V8的学习旅程吧