JavaScript学习 之 传值or传引用

目录

引言

JavaScript的函数参数到底传的是个啥?

有三种看法:

传值

传引用

基础类型传值 对象类型传引用

传值?

首先我们来看看是不是传值

// 例子1
function change_list(orig_list) {
    new_list = orig_list;
    new_list.push('new');
    return new_list;
}

orig_list = ['old'];
new_list = change_list(orig_list);

console.log('orig list: ' + orig_list);
console.log('new list: ' + new_list);

打印如下:

orig list: old,new
new list: old,new

由于orig_list都和new_list都发生了变化 因此

JavaScript不是传值 更像是传引用

传引用?

那到底是不是传引用呢? 我们来看看下面的这个例子

// 例子2
function inc(n) {
    n = n + 1;
    console.log('[in] n = ' + n);
}

n = 1;
inc(n);
console.log('[out] n = ' + n);

打印如下:

[in] n = 2
[out] n = 1

由于inc函数里和函数外的n并不一致 因此

JavaScript不全是传引用 有时也传值

基础类型传值 对象类型传引用?

看到这里机智的你 已经发现了"正确"的答案: 基础类型传值 对象类型传引用

关于JavaScript类型的更多介绍请参考JavaScript学习 之 类型

按照上面的两个例子 这个答案看起来确实是对的 那么是不是真的是这样呢? 来看下面的例子

// 例子3
function change_me(orig_list) {
    new_list = orig_list;
    if (new_list.length < 3) {
        new_list = [1000];
    } else {
        new_list = new_list.push(1000);
    }
}

var orig_list = [1];
change_me(orig_list);
console.log(orig_list);

orig_list = [1, 2, 3];
change_me(orig_list);
console.log(orig_list);

按照上述答案 此时orig_list是对象类型所以传引用 那么在调用change_me之后 期望的打印结果如下

// 期望的打印结果
[ 1000 ]
[ 1, 2, 3, 100]

那么实际的打印结果是否如期望的那样呢 使用babel-node执行该文件后 实际的打印结果如下

// 实际的打印结果
[ 1 ]
[ 1, 2, 3, 100]

关于babel-node的更多介绍请参考JavaScript学习 之 版本

同时是对象类型 为什么会这样呢?

第一次像是传值

第二次像是传引用

看来这种解释也是不对的 那JavaScript的参数到底是传得啥呢? 我已经晕了

传共享!

正确的表述应该是:

传共享(call-by-sharing)

当然 也可以说是传对象(call-by-object)或传对象的共享(call-by-object-sharing)

关于call-by-sharing的更多解释请参考这里

但是 什么叫做传共享 这个概念完全没听过啊!

首先 来看看例子1

传入的orig_list 在push操作之后 函数外的orig_list也被修改了

接着 再看看例子2

传入的n 在"n = n + 1"操作之后 函数里n的值为2 而函数外n的值仍然为1

最后 来看看例子3

传入的orig_list 在赋值新的对象时 函数外的orig_list并没有修改 而push操作时 函数外的origi_list会被修改

因此 我们可以这样理解传共享

  • 对对象进行修改时 调用者和被调用者之间共享这个对象 表现出来就像传引用

  • 对不可变的基本类型进行修改或者给对象赋值新的对象时 调用者和被调用者引用的已经不是同一个对象 表现出来就像传值

如果你觉得拗口, 那我只能说: 回头再看一遍例子吧!

小结

JavScript这种区分不可变 / 重新赋值和可变对象的做法 其实也是很多编程语言采取的一种通用做法

例如Python也有类似JavaScript的参数传递方式 详见Python函数参数是传值还是传引用?

这样设计的目的 我认为是为了提高效率 包括: 对象分配和内存使用的效率

当只是对可变对象进行修改 那么就不必分配新的对象 共享同一个共享 表现出来就像传引用

当需要修改不可变对象或者赋值新的对象 那么不得不分配新的对象 不再共享同一个对象

用一句话描述就是:

引用优先 按需分配

不知道机智的你 是如何理解JavaScript的参数传递的呢? 希望读者也分享你的观点和依据 我们一起讨论和完善对JavaScript参数传递的认识

参考

更多文章, 请支持我的个人博客

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容