博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
图解JavaScript的参数传递
阅读量:6327 次
发布时间:2019-06-22

本文共 1736 字,大约阅读时间需要 5 分钟。

写在最前

本次尝试通过流程图的形式并结合两个例子来重新理解一下JavaScript中的参数传递。

欢迎关注,不定期更新中——

参数到底如何传递?

借用红宝书的一句话:

ECMAScript中所有函数的参数都是按值传递的

这个值如果是简单类型,那么就是其本身。如果是引用类型也就是对象传递的就是指向这个对象的地址。故我们可以认为参数传递全部都是值传递,那么具体怎么理解呢?看下例子:

第一个例子

var obj = {    n: 1};function foo(data) {    data = 2;    console.log(data); //2}foo(obj);console.log(obj.n) // 1

先不说为什么原因,我们就通过画图的方式来走一遍流程,我相信应该就能理解其中的参数传递了。切记传递引用类型传递的是指针!

image
首先执行var obj = {n: 1}; ,可以看作在栈的001地址中存入了一个指向{n:1}的指针*p

接下来为声明function foo 此时会创建函数执行上下文,产生一个变量对象,其中声明了形参data,由于函数没有执行,当前值为undefined。我们记data地址为022。关于更多变量对象的知识可以参考冴羽老师的这篇,本文不深入研究关于AO相关,你只需要知道在声明这个函数的时候里面的形参已经被创建出来了。

执行foo(obj) 其中会进行参数传递,其中将obj中存储的*p拷贝给处在022地址的data,那么此时它们就指向了同一个对象,如果某一个变量更改了n的值,另一个变量中n的值也会更改,因为其中保存的是指针。

进入函数内部,顺序执行data = 2;此时002地址存储了基本类型值,则直接存储在栈中,从而与堆中的{n:1}失去了联系。从而打印console.log(data) // 2 ,最后发现初始开辟的{n:1}对象没有过更改,故而 console.log(obj.n) // 1仍然打印1。

第二个例子

var obj = {n:1};(function(obj){  console.log(obj.n); //1  obj.n=3;  var obj = {n:2};  console.log(obj.n) //2})(obj);console.log(obj.n) //3

整体来看这个例子中出现了同名覆盖的问题。不太了解代码如何执行的流程,可能会因为同名的关系而有些混乱,不过没关系。只要按照上一个例子的流程图中的执行过程,一定可以得出正确的结果。

声明变量obj,地址为011其中存入指向{n:1}的指针*p

声明函数,虽然同为obj变量名,但是形参obj为AO中的属性,不会与全局造成覆盖,其拥有新的地址记作022,在未执行前其值为undefined。

函数立即执行,此时将全局obj赋值给形参obj,我们忽略这个重复命名的问题,其实就是将011中的 指针*p拷贝了一份给了022。同时执行第一个console.log(obj.n)结果即为1。

执行obj.n=3,此时为函数的形参即022中的obj来改变了对象内n的值。

最关键的一步var obj = {n:2}; 由于对象命名的关系可能很多童鞋就会有点懵,但依然按照同样的方式来分析即可,由于使用了var那么就是新声明一个对象,从而会在栈中压入新的地址记作033,其中存入了新的指针指向了新的对象{n:2}。从而之后打印的console.log(obj.n)结果则应是新开辟的对象中的n的值。

最后打印 console.log(obj.n) //3很显然,全局的对象有过一次更改其值为3。

小结

至此我们走完了上述两段代码涉及变量的所有“心路历程”,由于作者不是科班出身,这个图中对于堆栈以及变量重名的描述可能不是非常的准确,有差错的地方还望不吝赐教~重点是能理解我希望表达的意思就好。总的来说关键点就在于传参的过程中存在一次值的拷贝,同时如果赋值对象是引用类型传入的是指针,明白这两点之后再加上之前流程图的分析相信再遇到类似的问题都可以有较为一致的思路了。

最后

惯例po,不定时更新中——

有问题欢迎在issues下交流。

转载地址:http://wnwoa.baihongyu.com/

你可能感兴趣的文章
Java设置以及获取JavaBean私有属性进阶
查看>>
json,serialize,msgpack比较
查看>>
《House of Cards》观后感
查看>>
android性能调优之traceview的使用
查看>>
微信公众平台开发文档 客服接口
查看>>
Sortable – 简单灵活的 JavaScript 拖放排序插件
查看>>
[转]SSIS中OLE DB Source中如何执行Store Procedure 以得到源数据
查看>>
Laravel资料
查看>>
BeanUtils使用
查看>>
Unable to locate secure storage module异常的解决方案
查看>>
大型网站架构 借鉴(1)
查看>>
AsyncHttpClient的连接池使用逻辑
查看>>
C++函数模板
查看>>
Centos安装Memcache
查看>>
使用srvany.exe将程序安装成windows服务的详细教程
查看>>
Sftp和ftp 差别、工作原理等(汇总ing)
查看>>
Linux 常用命令大全
查看>>
经常使用算法之动态规划法
查看>>
OpenStack Dashboard
查看>>
浏览器兼容性问题解决方式
查看>>