用一个例子理解JS函数的底层处理机制

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

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

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,变量对象的一种,类似全局上下文中的全局变量)
  4. 内部代码执行之前发生的事
    • 初始化作用域链scope-chain <EC(fn1),EC(G)>,链的两头是 <当前自己的私有上下文,函数的作用域(创建函数的时候所在的上下文)>,链的右侧也叫当前上下文的'上级上下文'
    • 初始化this
    • 初始化argument
    • 在当前上下文中,声明一个形参变量,并且把传递的实参值赋值给它
    • 变量提升
  5. 进栈执行代码
  6. 出栈释放

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

接着说说上面第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)
(0)

相关推荐

  • 深入理解JavaScript作用域和作用域链

    目录 前言 作用域(Scope) 1.什么是作用域 2.全局作用域和函数作用域 3.块级作用域 作用域链 1.什么是自由变量 2.什么是作用域链 3.关于自由变量的取值 作用域与执行上下文 解释阶段: ...

  • 闭包理解

    闭包 刚学过的闭包,分享一下闭包的理解 1.什么是闭包? 闭包就是一个函数,也可以说闭包是一个引用关系,可以理解为一个作用域可以访问另一个函数的局部变量. 代码: function fn() { va ...

  • js 作用域和作用域链

    一.作用域 在 Javascript 中,作用域分为 全局作用域 和 函数作用域 全局作用域: 代码在程序的任何地方都能被访问,window 对象的内置属性都拥有全局作用域. 函数作用域: 在固定的代 ...

  • JavaScript 之 作用域

    学习目标:能够说出Javascript的两种作用域 能够区分全局变量和局部变量 能够说出如何在作用域链中查找变量的值 1.作用域 <script> //1.javaScript作用域:就是 ...

  • javascript执行上下文、作用域与闭包(第六篇)

    终于讲到闭包了,当你在百度上搜索闭包时,你会被搜索出来的结果吓一跳,我的天,为什么说得都不一样?直到把所有的解释都看过了,我就只想说一句,到底谁说的是对的- 在这么多的不同解释里,我认真思考了很久,到 ...

  • C语言变量作用域以及栈,堆困惑!!!!

    https://m.toutiao.com/is/Jn8m7wr/ 这几天遇到一段很困惑的代码.用函数内部变量地址做返回值时候,没法正确返回变量a地址,返回的是OX0如下: 我在函数内部再加一个指针q ...

  • 《征服C指针》摘录

    一 3.3.2 "左值"是什么--变量的两张面孔   假设有下面这样一个声明: int hoge; 因为此时 hoge 是 int 类型,所以,只要是可以写 int 类型的值的地方 ...

  • 一个简单例子理解贝叶斯公式

    一个简单例子理解贝叶斯公式

  • js函数封装实现一个倒计时效果

    js函数封装实现一个倒计时效果

  • 深入理解js立即执行函数

    看过jQuery源码的人应该知道,jQuery开篇用的就是立即执行函数.立即执行函数常用于第三方库,好处在于隔离作用域,任何一个第三方库都会存在大量的变量和函数,为了避免变量污染(命名冲突),开发者们 ...

  • 一个反爬 JS 逆向分析的例子

    定位加密点 在某网站中进行登录请求: 简单抓下包,点击登录按钮之后,可以在浏览器的控制台中看到相关的请求: 接着往下拉,可以看到 POST 请求的参数信息: 从中可以看出,除了 username 中的 ...

  • 一个例子让你看清线程调度的随机性

    线程调度的几个基本知识点 多线程并发执行时有很多同学捋不清楚调度的随机性会导致哪些问题,要知道如果访问临界资源不加锁会导致一些突发情况发生甚至死锁. 关于线程调度,需要深刻了解以下几个基础知识点: 调 ...

  • 巴菲特‎举‎过一个例子:左轮手枪‎可装6...

    巴菲特‎举‎过一个例子:左轮手枪‎可装6‎颗子弹,如果只装1颗‎,让你对着自‎己‎的脑袋开‎枪,给100‎万美元,你干不干? 有人不‎干,因‎为一旦失败,代价过于‎惨‎重.‎也有人会干,因为冒着‎六分 ...

  • 一个复杂嵌套的函数公式是怎么一步一步写出来的

    经常有人问我,看你们公式写的好复杂好长,写起来好简单 是怎么写的? 我--也--不知道呀!       其实真的不好讲,也不知道从何讲起,但是小编还是试着来说说吧! 正好今天看到一位网友的问题,算是一 ...

  • 父母怎样教育好一个男孩? 理解男孩的思维方式, 教育出优秀的男孩

    作为父母,如何才能教育好一个男孩呢? 点击加载图片 在<这样养育男孩>一书中有句这样的话: 我们希望男孩能成为优秀的男人,而我们能够帮助他们成为的人,其中,理解他们是关键. 因此,父母要想 ...

  • 理解VLOOKUP函数第4个参数的真实含义

    VLOOKUP函数是大家最喜欢使用的函数之一,能够帮助我们实现很多任务.这里,让我们回过头来从细节处着手,再次深入探究VLOOKUP函数,进一步熟练掌握这个常用的函数. 一键直达 >> E ...