几个例子理解不一样数据类型的堆栈内存处理

若有错误烦请指正

js代码的运行环境

  • 浏览器 内核(引擎)
  • node
  • webview(hybrid,嵌入到手机app里面,在app里面运行)
  • ...

下面经过几个例子理解不一样数据类型的堆栈内存处理node

js如何运行(示例1)

var a = 12;
var b = a;
b = 13;
console.log(a);

浏览器可以运行js代码,是由于浏览器会在计算机内存中分配出一块内存,用来供代码执行,这块内存叫栈内存,也叫Stack,或者ECStack(Execution Context Stack)执行环境栈web

为了区分是哪一个区域(全局或者函数等)下的代码执行,会产生一个执行上下文(EC : Execution Context)。所谓执行上下文,实际上是一个抽象的概念,简单来理解就是代码执行区域的划分。浏览器

在全局环境下会产生 EC(G) :Execution Context (golbal) 全局执行上下文,其中VO(G)全局变量对象(Varibale Object)存储全局执行上下文声明的变量,而后进入栈内存执行。app

声明变量的步骤

接着开始执行,先声明变量,声明变量有三步函数

var [变量] = [值]spa

  1. 先建立值(执行等号右边)3d

    • 基本数据类型是直接存储在栈内存当中
    • 引用类型的值,都是开辟一个单独的内存空间(堆内存Heap)存储信息
  2. 声明变量 declare指针

    • 存放到当前上下文的变量对象中(VO/AO)
  3. 定义(赋值)变量:让变量和值关联到一块儿,也就是所谓的赋值操做,也叫定义(defined)或指针指向code

    • 因此var n; //默认值是undefined 未定义

因此var a = 12步骤是对象

  1. 在内存中开辟空间,存储12
  2. 声明变量a
  3. 12赋值给a

var b = a处理是

由于右侧的a不是值,因此不须要第一步,不须要在栈里面开辟空间,直接进行第二步声明,而后执行第三步,关联到12值(指针)

b=13的处理步骤是

  1. 在内存中开辟空间,存储13
  2. 由于b已经在当前上下文的变量对象中,因此不须要第二步声明
  3. 13赋值给b

结果:

整体执行逻辑:

js如何运行(示例2)

var a = {n: 12};
var b = a;
b['n'] = 13;
console.log(a.n);

当第一步建立的值是一个引用类型的值时候,值就无法直接存到栈里(没有这么大的空间)。当建立引用类型值的时候,会进行如下处理

  • 在计算机内存中分配一个单独的内存出来(堆内存 Heap)
  • 这块堆内存有一个16进制的地址用来寻找
  • 把对象中的键值对分别存储到堆内存当中
  • 把堆内存地址放置到栈中,供变量调用

这就是第一步,建立值的过程

第二步声明。第三部赋值,将16进制的地址赋值给变量

var b = a;时,由于a为 变量,因此不须要建立值,接着声明b,最后赋值,将栈中a指向的地址也一样赋值给b,让b也指向那个16进制地址

b['n'] = 13运行原理:

b['n'] = 13属于对象的成员访问

  • b首先基于地址0x000000找到堆内存
  • 把堆内存中成员为n的值改成13
  • console.log(a.n)也属于成员访问 因此输出13

顺序以下

总结:基本数据类型和引用数据类型的区别?
基本类型的值直接存储在栈内存当中,直接按照值操做,引用数据类型值是开辟单独的堆内存存储信息的,将堆内存的地址存在栈当中,操做的都是引用地址

js如何运行(示例3)

var a = {n: 12};
var b = a;
b = {n: 13};
console.log(a.n);

当到b = {n: 13};

新开辟一个堆内存{n: 13},将地址指向b

GO global object 全局对象

不一样于VO,VO(G)是全局变量对象,存储当前上下文声明的变量的

GO global object 全局对象,是加载页面默认造成的。在浏览器中,加载页面时,在全局上下文中会默认定义一个叫window的对象,其中有setTimeout,setInterval等供js调用的属性和方法


注意区分VO和GO

JavaScript 中有一个特殊的对象,称为 全局对象(Global Object),它及其全部属性均可以在程序的任何地方访问,即全局变量
在浏览器 JavaScript 中,一般 window 是全局对象, 而 Node。js 中的全局对象是 global,全部全局变量(除了 global 自己之外)都是 global 对象的属性。
在 Node。js 咱们能够直接访问到 global 的属性,而不须要在应用中包含它。
globalprocess__filename__dirname

js如何运行(示例4)

var a = {n: 1};
var b = a;
a.x = a = {n: 2};
console.log(a.x);
console.log(b);
var a = {n: 1};
var b = a;

前两行代码执行以下,不在阐述:

要想理解a.x = a = {n: 2};

简单说一下运算符优先级

var a=12,b=13;

至关于

var a=12
var b=13
var a=b=13

至关于

b=13
var a=b//或者var a=13

正常计算都是从右到左处理的(固然第一步仍是建立值)

可是无论是a.x=b=13仍是b=a.x=13
都要先计算a.x由于优先级比较高(成员访问的优先级为19,仅次于()运算,运算符优先级

a.x = a = {n: 2};运算步骤以下

  1. 开辟内存,假设地址为0x000001
  2. 将地址放入栈中
  3. a.x = 地址
  4. a = 地址

因此如今a指向0x000001,b指向0x000000,即
a{n:2}b{n:1,x:{n:2}}

结果:

一个变量只能够关联一个栈中的值,可是一个栈中的值,能够被多个变量关联

相关文章
相关标签/搜索