如有错误烦请指正

js代码的运行环境

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

下面通过几个例子理解不同数据类型的堆栈内存处理

js如何运行(示例1)

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

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

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

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

声明变量的步骤

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

var [变量] = [值]

  1. 先创建值(执行等号右边)

    • 基本数据类型是直接存储在栈内存当中
    • 引用类型的值,都是开辟一个单独的内存空间(堆内存Heap)存储信息
  2. 声明变量 declare
    • 存放到当前上下文的变量对象中(VO/AO)
  3. 定义(赋值)变量:让变量和值关联到一起,也就是所谓的赋值操作,也叫定义(defined)或指针指向
    • 所以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=12var b=13
var a=b=13

相当于

b=13var 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}}

结果:

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

js堆和栈的区别_几个例子理解不同数据类型的堆栈内存处理相关推荐

  1. 堆与栈的区别详细总结

    1.堆与栈的区别详细总结_Fighting++++的博客-CSDN博客_堆和栈的区别 2.堆和栈的区别 - 江雨牧 - 博客园 3.堆和栈的区别_内外皆秀的博客-CSDN博客_堆和栈的区别 4.一文读 ...

  2. mysql中堆和栈_堆和栈的区别

    在说堆和栈之前,我们先说一下JVM(虚拟机)内存的划分: Java程序在运行时都要开辟空间,任何软件在运行时都要在内存中开辟空间,Java虚拟机运行时也是要开辟空间的.JVM运行时在内存中开辟一片内存 ...

  3. 堆和栈最通俗的_堆与栈的区别

    一.预备知识-程序的内存分配 一个由c/C++编译的程序占用的内存分为以下几个部分 1.栈区(stack)- 由编译器自动分配释放 ,存放函数的参数值,局部变量的值等.其操作方式类似于数据结构 ...

  4. 堆和栈的区别(面试经验总结)

    C++中,内存分为5个区:堆.栈.自由存储区.全局/静态存储区和常量存储区. 栈:是由编译器在需要时自动分配,不需要时自动清除的变量存储区.通常存放局部变量.函数参数等. 堆:是由new分配的内存块, ...

  5. iOS 堆和栈的区别?

    前言 堆和栈是什么?有什么区别?是干嘛的?啥东西呀这是?别急,慢慢看下去 内存管理 移动设备的内存及其有限,每一个APP所能占用的内存是有限制的 (吐槽一下:iPhone6s还是16G起步,还好我也买 ...

  6. 堆和栈的区别 (转贴)

    非本人作也!因非常经典,所以收归旗下,与众人阅之!原作者不祥! 堆和栈的区别 一.预备知识-程序的内存分配 一个由c/C++编译的程序占用的内存分为以下几个部分 1.栈区(stack)- 由编译器自动 ...

  7. (009) java后台开发之堆和栈的区别

    转自:https://course.tianmaying.com/java-basic+object-usage# 堆和栈的区别 堆和栈都是Java中常用的存储结构,都是内存中存放数据的地方: 1.在 ...

  8. 什么是堆和栈以及区别详解

    堆 先进先出 存取速度慢 存储数组和对象 动态分配内存 都是实体 栈 先进后出 存取速度快 存储变量 什么是堆内存? 它的作用是用于存储java中的对象和数组 当我们new一个对象或者创建一个数组的时 ...

  9. 语言堆栈入门——堆和栈的区别

    原文:http://student.csdn.net/link.php?url=http://www.top-e.org%2Fjiaoshi%2Fhtml%2F427.html 格式和部分内容稍作修改 ...

最新文章

  1. Docker for windows挂载文件到Nginx目录踩坑小记
  2. vue创建二:引入本地图片
  3. 蓝桥杯-逆序对(java)
  4. Java多线程--使用future进行异步编程
  5. dpg learning 和q_深度学习和强化学习之间的差别有多大?
  6. 《大型网站技术架构》读书笔记
  7. shell蚂蚁森林_和“蚂蚁森林”的融合
  8. postman设置域名_Postman中文文档——设置
  9. 精美UI静态界面欣赏
  10. 数据库(MySQL)基础
  11. 网页如何展示PPT文档
  12. spring boot 使用过滤器过滤非法字符
  13. 图解 CMMI 2.0之(七)能力域
  14. mplayer的安装
  15. SIM800C我的风格-看流程AT
  16. 服务器文件自动备份工具
  17. 随机森林的简单学习记录
  18. PDF阅读器中如何删除附件
  19. 英语流利说 第38天
  20. 最详细的WordPress建站教程,什么都不会10分钟也可以

热门文章

  1. .NET西安社区 [拥抱开源,又见 .NET] 第二次活动简报
  2. .NET Core中的性能测试工具BenchmarkDotnet
  3. 开源的,跨平台的.NET机器学习框架ML.NET
  4. 用ASP.NET Core 2.1 建立规范的 REST API -- 缓存和并发
  5. 用 docker secrets 保存 appsettings.Production.json
  6. 微服务集成——《微服务设计》读书笔记
  7. .NET Core 2.0及.NET Standard 2.0
  8. 从真实项目中抠出来的设计模式——第一篇:策略模式
  9. 有效事件: 可取代数十种设计模式
  10. php实现目录及目录文件下的遍历