个人笔记,如有错误烦请指正

以下面代码的运行举例,一行行进行运行的解析

var x = [12, 23];
function fn(y) {y[0] = 100;y = [100];y[1] = 200;console.log(y);
}
fn(x);
console.log(x);

var x = [12, 23];运行如下

  1. 开辟堆内存,创建数组值,假设堆内存的地址为0x000000
  2. 声明变量x
  3. 赋值,即将x指向堆内存的地址0x000000

接着

function fn(y) {y[0] = 100;y = [100];y[1] = 200;console.log(y);
}

运行如下

上面这段代码是创建一个函数的过程。和创建一个变量类似:

  • 都是声明一个变量存储值
  • 步骤一样:第一步也是先创建一个堆内存,里面存的是函数,这个堆内存有一个地址,然后把地址赋值给变量
  • 声明:方式类似函数名也算变量,当我们声明函数function fn(y){...}时,相当于我们声明了一个变量,只不过值是函数。类似于var fn = function (y){...}的函数表达式。最终把一个函数作为值赋值给一个变量或者其他

所以创建一个函数,详细的执行顺序如下 1. 首先开辟一个堆内存,存储函数的值(假设地址为0x000001) - 对象的值在堆内存当中,存储的是它的键值对 - 函数的值在堆内存当中,存储的是它的代码,而且是以字符串的形式存储的 - 创建函数的时候,就声明了它的作用域(scope),scope值是当前创建函数的时候所处的上下文,即在哪个上下文中创建的,作用域就是谁

2. 接着声明变量fn,并且指向堆内存地址(假设为0x000001)

函数执行的步骤

fn(x);运行如下(函数执行的步骤)

  1. 函数执行时,永远传的是值,fn(x)传的是x的值,即x指向的0x000000堆内存地址
  2. fn(0x000000)形成一个全新的私有上下文EC(fn)
  3. 在函数形成的新的上下文中,生成一个私有化变量对象AO,用来存储当前上下文中声明的变量(Active Object活动对象,简称AO,变量对象的一种,类似全局上下文中的全局变量)
  1. 内部代码执行之前发生的事
  • 初始化作用域链scope-chain ,链的两头是 <当前自己的私有上下文,函数的作用域(创建函数的时候所在的上下文)>,链的右侧也叫当前上下文的'上级上下文'
  • 初始化this
  • 初始化argument
  • 在当前上下文中,声明一个形参变量,并且把传递的实参值赋值给它
  • 变量提升
  1. 进栈执行代码
  2. 出栈释放

函数进栈执行代码的详细步骤

接着说说上面第5步的详细步骤

把之前创建的函数,在堆内存中存储的代码字符串拿出来转换为代码一行一行的执行执行。

私有上下文中代码执行中如果遇到一个变量,首先看是否为自己的'私有变量',如果是'私有'的,则操作自己的,和外界没有必然的关系,如果不是自己私有的,则基于作用域链,向其上级上下文中查找,看是否为上级上下文中私有的,如果也不是,继续向上查找......一直找到EC(G)全局上下文为止,我们把这种查找过程称之为 作用域链查找机制 所以

y[0] = 100
y = [100]
y[1] = 200

是这样的执行的:

  1. y[0] = 100,y现在存储的内存地址为0x000000,所以修改这个地址下的值:

2. y = [100],出现了新的对象值,所以要开辟新的堆内存0x000002,创建值,赋值

3. y[1] = 200,将0x000002地址对应的对象的值进行修改

4. console.log(y);这个y就是0x000002对应的值[100,200]操作的是私有变量y

fn函数至此执行完毕。

接着执行外面的console.log(x);,此时的x为全局变量对象中的x,对应的地址为0x000000,所以直接进行输出[100,23]

如果fn执行完之后,继续执行其他函数,同样会经历这样的流程。形成新的上下文,进栈执行...如果函数非常多,会一直进栈,占内存会越来越大。所以为了优化,浏览器会默认做出很多回收机制

结果与总体流程

结果

总体流程图

其他说明点

js上下文分类

js上下文(哪一个区域下执行)分类

  • 全局上下文EC(G)
  • 函数执行形成的私有上下文
  • 块级私有上下文

什么是私有变量

私有变量是私有上下文声明的变量,包含

  1. 形参
  2. 代码执行的时候声明的变量var / let / const / function ... 注意与全局变量区别,没有直接关系,但是可能会存在一些间接关系,比如下面这段代码下全局变量x的值是0x000000,通过函数,将0x000000传给了私有变量y,y也是0x000000
function fn(y) {y[0] = 100;y = [100];y[1] = 200;console.log(y);
}
fn(x)

js cookies 存数组_用一个例子理解JS函数的底层处理机制相关推荐

  1. R语言使用lm构建线性回归模型、并将目标变量对数化(log10)实战:可视化模型预测输出与实际值对比图、可视化模型的残差、模型预测中系统误差的一个例子 、自定义函数计算R方指标和均方根误差RMSE

    R语言使用lm构建线性回归模型.并将目标变量对数化(log10)实战:可视化模型预测输出与实际值对比图.可视化模型的残差.模型预测中系统误差的一个例子 .自定义函数计算R方指标和均方根误差RMSE 目 ...

  2. 用实际例子理解回调函数(Calback)

    用实际例子理解回调函数(Calback) 在我们编码的过程中,调用和回调几乎无处不在,但是我对回调函数到底是怎样一回事并没有一个真正透彻的理解,最近我查找学习了一些资料,学到了很多. 我参考了一些知乎 ...

  3. js堆和栈的区别_几个例子理解不同数据类型的堆栈内存处理

    如有错误烦请指正 js代码的运行环境 浏览器 内核(引擎) node webview(hybrid,嵌入到手机app里面,在app里面运行) ... 下面通过几个例子理解不同数据类型的堆栈内存处理 j ...

  4. java长度为100的数组_产生一个int数组,长度为100,并向其中随机插入1-100,不重复...

    #define RANDOM(X) (rand() % X + 1) int main() { //标志数组 int book[] = {}; int result[] = {}; //默认的随机数种 ...

  5. java 最大子数组_求一个数组中子数组的最大和算法(Java实现)

    前几天在微信订阅号"待字闺中"中看到的一篇文章<小技巧求一个数组中子数组的最大和>,提供下Java的实现,并且在对题目做下小修改,本来打算直接在微信里直接回复,但是发现 ...

  6. js 浅拷贝直接赋值_第二十二篇 JS中浅拷贝的方法有哪些?

    重要:什么是拷贝?之前也写过类似的文章,大家可以看<理解js的深拷贝和浅拷贝原理和实现的方法> 首先来直观的感受一下什么是拷贝. let arr = [1, 2, 3]; let newA ...

  7. node js 写按键精灵_带有按键的Node.js Raw模式

    node js 写按键精灵 I find the stuff that people are doing with Node.js incredibly interesting.  You here ...

  8. js 多个定时器_从浏览器多进程到JS单线程,JS运行机制最全面的一次梳理(二)

    作者:撒网要见鱼   https://segmentfault.com/a/1190000012925872 本文接上篇 <从浏览器多进程到JS单线程,JS运行机制最全面的一次梳理(一)> ...

  9. form select multiple 某个字段是数组_你知道什么是Select函数吗?

    Select函数:允许进程指示内核等待多个事件中的任何一个发生,并只在有一个或多个事件发生或经历一段指定的时间后才唤醒它. select man手册 举个栗子,我们可以调用Select,告诉内核仅仅在 ...

最新文章

  1. LightHouse/归并排序
  2. Java并发编程之线程池及示例
  3. 【数据平台】Centos下仅CPU安装TensorFlow
  4. HttpServlet的doGet()和doPost()方法
  5. Opencv中IplImage的四字节对齐问题
  6. mybatis学习(43):一级缓存被刷新情况
  7. js字符串转数字(小数),数字转字符串
  8. js获取元素的方法与属性
  9. 解析Node.js v6.9.5官方文档的第一个例子的知识点
  10. node2vec代码实现及详细解析
  11. 神州数码交换机enable密码清除
  12. 利用Md2all的自定义CSS,给Markdown一个漂亮的排版
  13. ml-agents_使用ML-Agents的自玩功能来训练智能对手
  14. 【案例分析】服务器数据恢复
  15. pytorch、tensorflow之生成one-hot向量
  16. android身份证自动识别
  17. 三款过CE/FCC/SRCC认证USB接口双频WIFI模块
  18. 如何挑选一双合适的童鞋
  19. gb和gib的区别_GB和GIB的区别
  20. android tab 凸起,三星Tab S7使用一周后,为何说安卓平板仍大有作为?

热门文章

  1. spring实现IOC的思路和方法
  2. Spring源码(1)
  3. 使用response的outputstream
  4. GraphQL入门之Schema和类型规范
  5. SpringMVC异常处理之异常处理代码编写
  6. 函数作为返回值练习 作用域和作用域链及预解析 闭包 闭包小案例
  7. java.lang.unsatisfiedlinkerror:_java.lang.UnsatisfiedLinkError: 的问题
  8. flyme禁止系统更新_魅族Flyme更新8.1.2.3A:重要系统更新!
  9. 解决MySQL报错... right syntax to use near ‘password ‘XXX‘ at line 1...ERROR 1064 42000: You have an erro
  10. Java官方相关资源文件的获取教程