JavaScript8的知识点有哪些
这篇文章主要介绍"JavaScript8的知识点有哪些",在日常操作中,相信很多人在JavaScript8的知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答"JavaScript8的知识点有哪些"的疑惑有所帮助!接下来,请跟着小编一起来学习吧!
单线程
如果你是一个Node
开发者,你应该很熟悉V8
的单线程特性。一个 JS 执行上下文与线程数量成正比。
当然,V8
在后台管理操作系统线程机制。它可以与多个线程一起工作,因为它是一个复杂的软件,可以同时执行许多任务。
但是,V8
为每个 JavaScript 的执行上下文只创建一个单线程的环境。其余的都在V8
的控制之下。
想象一下 JavaScript 代码应该进行的函数调用堆栈。 JavaScript 的工作原理是将一个函数堆叠在另一个函数之上,遵循每个函数的插入/调用顺序。在到达每个函数的内容之前,我们无法知道它是否调用其他函数。如果发生这种情况,那么被调用的函数将被放在堆栈中调用者的后面。
例如,当涉及回调时,它们被放在堆栈的末尾。
管理这个堆栈组织和进程所需的内存是V8
的主要任务之一。
Ignition and TurboFan
自2017年5月发布的5.9版以来,V8
附带了一个新的JavaScript执行管道,它构建在V8
的解释器Ignition
之上。它还包括一个更新和更好的优化编译器-TurboFan
。
这些变化完全集中在整体性能上,以及 Google 开发人员在调整引擎以适应 JavaScript 领域带来的所有快速而显著的变化时所面临的困难。
从项目一开始,V8
的维护人员就一直在担心如何在 JavaScript 不断发展的同时,找到一种提高V8
性能的好方法。
Hidden Classes(隐藏类)
这是V8的另一个魔术。JavaScript 是一种动态语言。这意味着可以在执行期间添加、替换和删除新属性。例如,在Java这样的语言中,这是不可能的,在Java
中,所有的东西(类、方法、对象和变量)都必须在程序执行之前定义,并且在应用程序启动后不能动态更改。
由于它的特殊性质,JavaScript 解释器通常基于散列函数(hash算法)执行字典查找,以准确地知道这个变量或对象在内存中的分配位置。
这对最后一道工序来说代价很大。在其他语言中,当对象被创建时,它们接收一个地址(指针)作为其隐式属性之一。这样,我们就可以准确地知道它们在内存中的位置以及要分配多少空间。
对于 JavaScript,这是不可能的,因为我们无法映射出不存在的内容。这就是Hidden Classes
发挥作用的地方。
隐藏类与Java
中的类几乎相同:静态类和固定类具有唯一的地址来定位它们。然而,V8
并不是在程序执行之前执行,而是在运行过程中,每次对象结构发生"动态变化"时执行。
让我们看一个例子来说明问题。考虑以下代码片段:
function User(name, fone, address) { this.name = name this.phone = phone this.address = address}
在 JavaScript 基于原型的特性中,每次实例化一个新的用户对象时,假设:
var user = new User("John May", "+1 (555) 555-1234", "123 3rd Ave")
然后V8
创建一个新的隐藏类。我们称之为_User0
。
每个对象在内存中都有一个对其类表示的引用。它是类指针。此时,由于我们刚刚实例化了一个新对象,所以在内存中只创建了一个隐藏类。现在是空的。
当你在这个函数中执行第一行代码时,将在上一个基础上创建一个新的隐藏类,这次是_User1
它基本上是具有name
属性的User
的内存地址。在我们的示例中,我们没有使用仅将name
作为属性的user
,但每次这样做时,这就是V8
将作为引用加载的隐藏类。
name
属性被添加到内存缓冲区的偏移量 0,这意味着这将被视为最后顺序中的第一个属性。
V8
还将向_User0
隐藏类添加一个转换值。这有助于解释器理解:每次向User对象添加name属性时,必须处理从_User0
到_User1
的转换。
当调用函数中的第二行时,同样的过程再次发生,并创建一个新的隐藏类
你可以看到隐藏类跟踪堆栈。在由转换值维护的链中,一个隐藏类通向另一个。
属性添加的顺序决定了V8
将要创建多少个隐藏类。如果您更改我们所创建的代码段中的行的顺序,那么也将创建不同的隐藏类。这就是为什么有些开发人员试图保持重用隐藏类的顺序,从而减少开销。
Inline Caching(内联缓存)
这是JIT(Just-in-Time)编译器中非常常见的一个术语。它与隐藏类的概念直接相关。
例如,每当你调用一个函数,将一个对象作为参数传递时,V8会看到这个动作,然后想:"嗯,这个对象作为参数成功地传递了两次或更多次……为什么不把它存储在我的缓存中以备将来调用,而不是再次执行整个耗时的隐藏类验证过程?"
让我们回顾上一个例子:
function User(name, fone, address) { // Hidden class _User0 this.name = name // Hidden class _User1 this.phone = phone // Hidden class _User2 this.address = address // Hidden class _User3}
当我们将 User 对象的实例两次作为参数传递给函数后,V8
将跳转到隐藏类查找并直接转到偏移量的属性。这要快得多。
但是,请记住,如果更改函数中任何属性赋值的顺序,则会导致不同的隐藏类,因此V8
将无法使用内联缓存功能。
这是一个很好的例子,说明开发人员不应该避免更深入地了解引擎。相反,拥有这些知识将有助于代码更好地执行。
Garbage Collecting(垃圾回收)
你还记得我们提到过V8
在另一个线程中收集内存垃圾吗?这很有帮助,因为我们的程序执行不会受到影响。
V8使用众所周知的"标记和扫描"策略来收集内存中的旧对象。在这种策略中,GC扫描内存对象以"标记"它们以进行收集的阶段有点慢,因为这需要暂停代码执行。
但是,V8是递增的,也就是说,对于每个 GC 停顿,V8尝试标记尽可能多的对象。它使一切变得更快,因为在集合完成之前不需要停止整个执行。在大型应用程序中,性能的提高有很大的不同。
到此,关于"JavaScript8的知识点有哪些"的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注网站,小编会继续努力为大家带来更多实用的文章!