第4章 变量、作用域和内存---JS红宝书书摘系列笔记
一、基本类型和引用类型
ECMAScipt变量可能分为两种数据类型:基本类型和引用类型。
基本类型:指简单的数据段;包括Undefined、Null、Boolean、Number、String;可以操作保存在变量中值(栈内存),所以称为按值访问;不能添加属性。
引用类型:可能由多个值构成的对象;包括Arry、Object等;js不允许直接操作对象的内存(堆内存)空间,所以成为按引用访问;可以动态得添加/改变/删除引用类型值的属性和方法。
1.复制
1 var a=5;
2 var b=a;
3 console.log(a);//5
4 console.log(b);//5
5 a=3;
6 console.log(a);//3
7 console.log(b);//5
8 var arrA=[1,2,3];
9 var arrB=arrA;
10 console.log(arrA);//[1,2,3]
11 console.log(arrB);//[1,2,3]
12 arrA[0]='x';
13 console.log(arrA);//['x',2,3]
14 console.log(arrB);//['x',2,3]
上述代码中a、b为基本数据类型,arrA、arrB为引用类型。可以看出首先定义并初始化了变量a为5,再定义变量b,此时打印出来a和b都是5;改变a的值为3,再次打印,a为3,b为5。对于引用类型arrA以及arrB,进行类似的操作,会发现arrA和arrB的值保持一致。继续看如下代码
1 var arrC=[1,2,3];
2 var arrD=arrC;
3 console.log(arrC);//[1,2,3]
4 console.log(arrD);//[1,2,3]
5 arrC=['X','Y','Z'];
6 console.log(arrC);//['X','Y','Z']
7 console.log(arrD);//[1,2,3]
此时引用类型arrC、arrD的值并没有改变。这是因为基本类型是在变量对象上,而引用类型是保存在内存中。基本类型的变量是完全相互独立的,而引用类型变量的复制拷贝的实则上是一个指针,这个指针指向存储在内存中的对象。只要指针是指向改变的对象时,则通过这些指针访问的也会发生改变,如arrA、arrB。而arrC与arrD之所以前后没有保持一致,是因为arrC=['X','Y','Z']这个操作,是将arrC的指针指向了另一个对象,即['X','Y','Z']。而arrD的指针的指向并没有发生改变,仍然指向[1,2,3]。
2.传递参数
函数的参数传递是按值传递的,无论是基础类型还是引用类型。这一点书上说的很详细,可以参考书上P71页。
3.检测类型
typeof用来检测基本数据类型,instanceof用来检测引用类型。因为引用类型的值都是Object的实例,所以instanceof检测引用类型和object构造函数时返回true;反之instanceof检测基本类型时,由于基本类型不是对象,所以始终返回false。
二、执行环境及作用域
Web浏览器中,全局执行环境被认为是window对象,所以所有全局变量和函数都是作为window对象的属性和方法创建的,不过一般省略了window。个人感觉这一小节内容还是比较容易理解的,执行环境就只分为两种:全局和局部。
1.延长作用域链
两种方法:try-catch和with语句。由于witch语句会给性能带来影响,所以以前者为例,catch语句会在作用域链的前端添加一个变量对象A,其中包含的是被抛出错误对象的申明,即在catch语句内部,可以引用对象A的属性或方法。
2.JS语言没有块级作用域
由于访问局部变量不用向上搜索作用域链,所以访问局部变量要比访问全局变量要快。即使快的不明显,但是仍然要尽量避免使用全局变量,防止变量污染。
三、垃圾收集
JS具有自动垃圾收集机制,所需内存的分配以及无用内存的回收完全实现了自动管理。其原理是:垃圾收集器会按照固定的时间间隔周期性地找出那些不再继续使用的变量,然后释放其占用的内存。局部变量在函数执行完毕时会自动被销毁。
1.JS垃圾收集方式有两种:标记清除和引用计数。
前者最常用,简单来讲,垃圾收集器会给存储在内存中的所有变量加上标记,即进入执行环境的变量(占用内存)会被加上一个标记表示不会被清除;当变量要离开环境,执行完毕时,这些变量会再次被标记表示要被清除,此时环境中的变量已经访问不到这些变量。
引用计数是指跟踪记录每个值被引用的次数。当声明一个变量并将一个引用类型赋值给该变量时,其引用次数为1,如果同一个值又被赋给另一个变量,则该值的引用次数家1;相反,如果包含对这个值引用的变量又取到另外一个值,则引用次数减1.当内存中值的引用次数为0时,其占用的内存空间会被回收。通俗来讲,引用次数就是指向内存对象的指针的个数。可以类比前面的arrC和arrD,如下图:
1 var arrC=[1,2,3];
2 var arrD=arrC;
3 console.log(arrC);//[1,2,3]
4 console.log(arrD);//[1,2,3]
5 arrC=['X','Y','Z'];
6 console.log(arrC);//['X','Y','Z']
7 console.log(arrD);//[1,2,3]
首先内存中的值为[1,2,3],当声明arrC并将值[1,2,3]赋给arrC时,值的引用次数为1,即arrC的指针指向它。当将arrC的值赋给arrD时,值[1,2,3]的引用次数+1为2,此时arrC、arrD的指针都指向值[1,2,3]。接下来,将值['X','Y','Z']赋给arrC时,arrC的指针不再指向值[1,2,3],而是指向值['X','Y','Z'],此时arrD依旧指向值[1,2,3]。值[1,2,3]的引用次数减1为1。引用计数最大的弊端就是循环引用,即两个对象都包含指向对方的引用。
2.性能问题
垃圾收集器是周期运行的,时间间隔的确定会影响浏览器的性能。
3.管理内存
解除引用:将数据的值置为null,适用于大多数全局变量和全局对象的属性,其作用是让值脱离执行环境,以便垃圾收集器下次运行时将其回收。
第4章 变量、作用域和内存---JS红宝书书摘系列笔记的更多相关文章
- 第5章 引用类型---JS红宝书书摘系列笔记
在ECMAScript中,引用类型是一种数据结构,用于将数据和功能组织在一起,描述的是一类对象所具有的属性和方法.而对象是某个特定引用类型的实例. 一.Object类型 可以通过Object构造函数创 ...
- 读书笔记 - js高级程序设计 - 第四章 变量 作用域 和 内存问题
5种基本数据类型 可以直接对值操作 判断引用类型 var result = instanceof Array 执行环境 每个执行环境都有一个与之关联的变量对象,环境中定义的所有变量和函数都保存在这 ...
- javascript高级程序设计第3版——第4章 变量作用域以及内存
- JS红宝书笔记——第一章 JavaScript简介
1.JavaScript简史 Netscape公司决定开发一种客户端语言用来处理浏览器端简单的表单验证. Netscape公司派布兰登·艾奇(BrendanEich)为计划于1995年2月发布的Net ...
- js红宝书学习笔记(一)引用类型
一.引用类型 ECMAScript中,引用类型是一种数据结构称之为对象定义,,引用对象不同于传统面向对象语言所支持的类和接口等基本结构 创建Object 实例的两种方式: new操作符跟Object构 ...
- 第一百零六节,JavaScript变量作用域及内存
JavaScript变量作用域及内存 学习要点: 1.变量及作用域 2.内存问题 JavaScript的变量与其他语言的变量有很大区别.JavaScript变量是松散型的(不强制类型)本质,决定了它只 ...
- [转] VS2015中跑OpenGL红宝书第八版的第一章示例代码,运行
Ori Article Link OpenGL的东西快忘光了,把角落的第八版红宝书拿出来复习一下 从书中的地址下了个示例代码结果新系统(Win10+VS2015)各种跑不起来,懊恼之后在网上疯狂搜索资 ...
- js 变量 作用域及内存
由于Javascript是松散型的,所以其变量只是在特定时间用于保存特定值的一个名字而已,并不存在某个变量必须保存某种类型的值的规则,变量的值以及其数据类型都可以在脚本的声明周期内改变 一.基本类型与 ...
- js基础之--变量 作用域和内存问题
基本类型:Undefind Null Boolean Number String 引用类型: 对象 在操作对象时,实际上实在操作对象的引用而不是实际的对象.为此,引用类型的值是按引用访问的. 从一个变 ...
随机推荐
- EF之贪婪加载和延迟加载
这篇文章将讨论查询结果的控制 在使用EF(Entity Framework)的过程中,很多时候我们会进行查询的操作,因此知道哪些数据会被加载到内存当中就至关重要.在多多的情况下,你可能并并不需要加载全 ...
- SQL Server 盛宴系列 webcast
数据库学习站点,示例,视频 http://technet.microsoft.com/zh-cn/dd722671.aspx
- linux内核中ip,tcp等头的定义(转)
一.MAC帧头定义 /*数据帧定义,头14个字节,尾4个字节*/typedef struct _MAC_FRAME_HEADER{ char m_cDstMacAddress[6]; //目的m ...
- Flutter实战视频-移动电商-10.首页_FlutterSwiper轮播效果制作
10.首页_FlutterSwiper轮播效果制作 博客地址: https://jspang.com/post/FlutterShop.html#toc-5c2 flutter_swiper http ...
- <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">详解
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 关于Angular6版本升级和RXJS6新特性的讲解
从Angular5升级到Angular6, angular6相比较于angular5 总体变化不大,但是在RXJS上面却有一些变动,不得不说,虽然很讨厌break update但是RXJS6的新特性对 ...
- c# UserControl 如何在父窗口中显示Scroll
1. UserControl 不应该是Dock = Fill, 而应该使用 Anchor 2. 父窗口(或者父控件,或者容器)的AutoScroll = True 3.UserControl 加入容器 ...
- hdu 6319 Problem A. Ascending Rating (2018 Multi-University Training Contest 3)
#include <stdio.h> #include <iostream> #include <cstdlib> #include <cmath> # ...
- JSP | 基础 | Button跳转页面
<input type = "button" value = "登陆" onclick = "window.location.href = 'L ...
- 统一入口&DB代理思想
参考 https://mp.weixin.qq.com/s?__biz=MzI4NTA1MDEwNg==&mid=2650764278&idx=2&sn=b7801fb0780 ...