浅习一波JavaScript高级程序设计(第4版)p4

简介: JavaScript 高级程序设计第 4 版(后简称高程4),相较于第 3 版,增加了 ES6 至 ES10 的全新内容,删除了旧版过时的内容,并在原有基础上充实了更加翔实的内容。

image.png

JavaScript 高级程序设计第 4 版(后简称高程4),相较于第 3 版,增加了 ES6 至 ES10 的全新内容,删除了旧版过时的内容,并在原有基础上充实了更加翔实的内容。

中文译版于 2020 年发售,妥妥的“新鲜出炉”,你要是问本瓜:当今学 JavaScript 哪家强,我只能说:红宝书第 4 版最在行。


于是乎,借着更文契机,本瓜将开启一个小系列,带你重看一遍高级程序设计4(先前只是跳着跳着看),将抽取精华,用最简单的话解释核心点、尽量把握全局、快速过一遍的同时,记录与工友们分享~~


正文



第四章:变量、作用域与内存,这部分就比较核心了,需要重点突破突破。

首先讲到 ECMAScript 变量最大的两个特点:原始值和引用值


当我们在把一个值赋给变量时,JavaScript 引擎必须确定这个值是原始值还是引用值。

原始值有 6 个,前文提过:Undefined、Null、Boolean、Number、String 和 Symbol,保存原始值的变量是按值(byvalue)访问的;


image.png


而引用值则是对象,在操作对象时,实际上操作的是对该对象的引用(reference)而非实际的对象本身。

image.png


基本类型值在内存中占据固定大小,直接存储在【栈内存】中的数据 引用数据类型;

引用类型在【栈中存储了指针】,这个指针指向堆内存中的地址,【真实的数据存放在堆内存】里。


在很多语言中,字符串是使用对象表示的,是引用类型。ECMAScript打破了这个惯例。


接着,说道一个重点:传递参数!!


书上是这样说的:ECMAScript 中所有函数的参数都是按值传递的。

本瓜以为,这样解释,会导致歧义。应该说:ECMAScript 中所有函数的参数传递就跟参数变量的复制是一样的,但函数内部声明的变量,都是局域的; 怎么理解?


原始值,很好理解,即函数内部的操作,不会影响外部值

function addTen(num) {
 num += 10;
 return num;
}
let count = 20;
let result = addTen(count);
console.log(count); // 20,没有变化
console.log(result); // 30


函数内部的修改外部对象,竟然影响了,因为对象的复制就是按引用值来的;

function setName(obj) {
 obj.name = "Nicholas";
}
let person = new Object();
setName(person);
console.log(person.name); // "Nicholas"


不过,特别注意的是:像下面的代码中,重新声明的 obj (或者重定义),是不会影响外部对象的;

function setName(obj) {
 obj.name = "Nicholas";
 obj = new Object();
 obj.name = "Greg";
}
let person = new Object();
setName(person);
console.log(person.name); // "Nicholas"


如果函数结束了,函数内部定义的 obj,就会被销毁。

这里理解起来确实有点麻烦 QAQ


记住:

函数外的值会被复制到函数内部的参数中,就像从一个变量复制到另一个变量一样;如果是原始值,那么就跟原始值变量的复制一样,如果是引用值,那么就跟引用值变量的复制一样。

当对象在函数内部被重写时,它变成了一个指向本地对象的指针,不会对外部变量造成影响,本地对象在函数执行结束时就被销毁了。


然后,顺嘴带过:instanceof 检测引用值,这个前文说过,不作赘述


接着,第四章来到了:执行上下文与作用域, 这个重要,理解起来不难,这里也只是浅讲概念:


变量或函数的上下文决定了它们可以访问哪些数据,以及它们的行为。

上下文中的代码在执行的时候,会创建变量对象的一个作用域链(scope chain)。这个作用域链决定了各级上下文中的代码在访问变量和函数时的顺序。


然后,谈到:变量提升(前文说过)


接着,谈及:垃圾回收


这个是涉及原理性的知识,平常业务中很少接触,但又很重要!

JavaScript 通过自动内存管理实现内存分配和闲置资源回收。基本思路很简单:确定哪个变量不会再使用,然后释放它占用的内存。 这个过程是周期性的,即垃圾回收程序每隔一定时间(或者说在代码执行过程中某个预定的收集时间)就会自动运行。


  1. JavaScript 最常用的垃圾回收策略是标记清理
    当变量进入上下文,比如在函数内部声明一个变量时,这个变量会被加上存在于上下文中的标记。而在上下文中的变量,逻辑上讲,永远不应该释放它们的内存(闭包) ,因为只要上下文中的代码在运行,就有可能用到它们。当变量离开上下文时,也会被加上离开上下文的标记。
    垃圾回收程序运行的时候,会标记内存中存储的所有变量(标记方法有很多种)。然后,它会将所有在上下文中的变量,以及被在上下文中的变量引用的变量的标记去掉。在此之后再被加上标记的变量就是待删除的了,原因是任何在上下文中的变量都访问不到它们了。随后垃圾回收程序做一次内存清理,销毁带标记的所有值并收回它们的内存。
  2. JavaScript 另一种没那么常用的垃圾回收策略是引用计数
    其思路是对每个值都记录它被引用的次数。声明变量并给它赋一个引用值时,这个值的引用数为 1。如果同一个值又被赋给另一个变量,那么引用数加 1。
    类似地,如果保存对该值引用的变量被其他值给覆盖了,那么引用数减 1。当一个值的引用数为 0 时,就说明没办法再访问到这个值了,因此可以安全地收回其内存了。


不同的浏览器垃圾回收机制是不一样的,我们没办法悉数了解后根据它们的策略来制定代码方案,只能尽量保证:变量不需要了,就请尽快的回收它,比如设置成 null


将内存占用量保持在一个较小的值可以让页面性能更好。优化内存占用的最佳手段就是保证在执行代码时只保存必要的数据。如果数据不再必要,那么把它设置为 null,从而释放其引用。这也可以叫作解除引用。这个建议最适合全局变量和全局对象的属性。局部变量在超出作用域后会被自动解除引用;


还有,闭包会造成内存泄漏,这些以后专题谈闭包再说吧。。。


小结



前文提到:我们提倡使用:const>let>var,从内存管理、垃圾回收的角度也能解释,因为 const 和 let 都以块(而非函数)为作用域,所以相比于使用 var,使用这两个关键字可能会更早地让垃圾回 收程序介入,尽早回收应该回收的内存。

看,很多东西都是相通的。


image.png


相关文章
|
8月前
|
中间件 Go 数据库
Go开发者必读:Gin框架的实战技巧与最佳实践
在当今快速发展的互联网时代,Web开发的需求日益增长。Go语言以其简洁、高效、并发性强的特点,成为了开发者们的首选。而在Go语言的众多Web框架中,Gin无疑是其中的佼佼者。本文将深入探讨Gin框架的特性、优势以及如何利用Gin构建高性能的Web应用。
|
存储 移动开发 负载均衡
晨兴夜寐:这一次,彻底搞懂Cookie,LocalStorage,SessionStorage
晨兴夜寐:这一次,彻底搞懂Cookie,LocalStorage,SessionStorage
1344 0
|
JavaScript 前端开发
《JavaScript高级程序设计》__ 语言基础(上)(1)
前言 大家好,我是HoMeTown,web领域有一本神书大家应该都有看过,这本书我看过两遍,但是每次看都是粗粗的略过一些重要的知识点,甚至一些面试过程中的问题,在这本书里都能找到答案。
158 3
|
前端开发 JavaScript
学习JavaScript笔记
学习JavaScript笔记
82 0
|
8月前
|
机器学习/深度学习 人工智能 小程序
一文秒懂ChatGPT官方提示词最佳实践(中)(一)
一文秒懂ChatGPT官方提示词最佳实践(中)
一文秒懂ChatGPT官方提示词最佳实践(中)(一)
|
8月前
|
缓存 NoSQL 关系型数据库
MySQL与Redis的默契协作:解析数据一致性难题与解决方案
MySQL与Redis的默契协作:解析数据一致性难题与解决方案
387 0
MySQL与Redis的默契协作:解析数据一致性难题与解决方案
|
8月前
|
数据安全/隐私保护 iOS开发 开发者
Uniapp 最新版 iOS 打包详细步骤详解
Uniapp 最新版 iOS 打包详细步骤详解
|
8月前
|
数据可视化 JavaScript 前端开发
D3.js的交互式图表和可视化效果
在当今数据爆炸的时代,有一个强大的工具可以帮助我们更好地理解和使用数据:D3.js。D3.js是一个流行的JavaScript库,用于创建交互式图表和可视化效果。本文将介绍D3.js的基本特性以及如何使用它来创建高质量的数据可视化。
|
JavaScript 算法 编译器
vue2和vue3diff算法的区别?
vue2和vue3diff算法的区别?
321 0