摘要:按引用傳遞時(shí),函數(shù)的形參接收實(shí)參的隱式引用,而不再是副本。探究值的傳遞方式的基本類型,是按值傳遞的。但這樣是否說明的對象是按引用傳遞的呢我們再看下面的例子仍然是并未被修改為如果是按引用傳遞,修改形參的值,應(yīng)該影響到實(shí)參才對。
最近遇到個(gè)有趣的問題:“JS中的值是按值傳遞,還是按引用傳遞呢?”
在分析這個(gè)問題之前,我們需了解什么是按值傳遞(call by value),什么是按引用傳遞(call by reference)。在計(jì)算機(jī)科學(xué)里,這個(gè)部分叫求值策略(Evaluation Strategy)。它決定變量之間、函數(shù)調(diào)用時(shí)實(shí)參和形參之間值是如何傳遞的。
按值傳遞 VS. 按引用傳遞按值傳遞(call by value)是最常用的求值策略:函數(shù)的形參是被調(diào)用時(shí)所傳實(shí)參的副本。修改形參的值并不會(huì)影響實(shí)參。
按引用傳遞(call by reference)時(shí),函數(shù)的形參接收實(shí)參的隱式引用,而不再是副本。這意味著函數(shù)形參的值如果被修改,實(shí)參也會(huì)被修改。同時(shí)兩者指向相同的值。
按引用傳遞會(huì)使函數(shù)調(diào)用的追蹤更加困難,有時(shí)也會(huì)引起一些微妙的BUG。
按值傳遞由于每次都需要克隆副本,對一些復(fù)雜類型,性能較低。兩種傳值方式都有各自的問題。
探究JS值的傳遞方式JS的基本類型,是按值傳遞的。
var a = 1; function foo(x) { x = 2; } foo(a); console.log(a); // 仍為1, 未受x = 2賦值所影響
再來看對象:
var obj = {x : 1}; function foo(o) { o.x = 3; } foo(obj); console.log(obj.x); // 3, 被修改了!
說明o和obj是同一個(gè)對象,o不是obj的副本。所以不是按值傳遞。 但這樣是否說明JS的對象是按引用傳遞的呢?
我們再看下面的例子:
var obj = {x : 1}; function foo(o) { o = 100; } foo(obj); console.log(obj.x); // 仍然是1, obj并未被修改為100.
如果是按引用傳遞,修改形參o的值,應(yīng)該影響到實(shí)參才對。但這里修改o的值并未影響obj。 因此JS中的對象并不是按引用傳遞。那么究竟對象的值在JS中如何傳遞的呢?
按共享傳遞 call by sharing準(zhǔn)確的說,JS中的基本類型按值傳遞,對象類型按共享傳遞的(call by sharing,也叫按對象傳遞、按對象共享傳遞)。最早由Barbara Liskov. 在1974年的GLU語言中提出。該求值策略被用于Python、Java、Ruby、JS等多種語言。
該策略的重點(diǎn)是:調(diào)用函數(shù)傳參時(shí),函數(shù)接受對象實(shí)參引用的副本(既不是按值傳遞的對象副本,也不是按引用傳遞的隱式引用)。 它和按引用傳遞的不同在于:在共享傳遞中對函數(shù)形參的賦值,不會(huì)影響實(shí)參的值。如下面例子中,不可以通過修改形參o的值,來修改obj的值。
var obj = {x : 1}; function foo(o) { o = 100; } foo(obj); console.log(obj.x); // 仍然是1, obj并未被修改為100.
然而,雖然引用是副本,引用的對象是相同的。它們共享相同的對象,所以修改形參對象的屬性值,也會(huì)影響到實(shí)參的屬性值。
var obj = {x : 1}; function foo(o) { o.x = 3; } foo(obj); console.log(obj.x); // 3, 被修改了!
對于對象類型,由于對象是可變(mutable)的,修改對象本身會(huì)影響到共享這個(gè)對象的引用和引用副本。而對于基本類型,由于它們都是不可變的(immutable),按共享傳遞與按值傳遞(call by value)沒有任何區(qū)別,所以說JS基本類型既符合按值傳遞,也符合按共享傳遞。
var a = 1; // 1是number類型,不可變 var b = a; b = 6;
據(jù)按共享傳遞的求值策略,a和b是兩個(gè)不同的引用(b是a的引用副本),但引用相同的值。由于這里的基本類型數(shù)字1不可變,所以這里說按值傳遞、按共享傳遞沒有任何區(qū)別。
基本類型的不可變(immutable)性質(zhì)基本類型是不可變的(immutable),只有對象是可變的(mutable). 例如數(shù)字值100, 布爾值true, false,修改這些值(例如把1變成3, 把true變成100)并沒有什么意義。比較容易誤解的,是JS中的string。有時(shí)我們會(huì)嘗試“改變”字符串的內(nèi)容,但在JS中,任何看似對string值的”修改”操作,實(shí)際都是創(chuàng)建新的string值。
var str = "abc"; str[0]; // "a" str[0] = "d"; str; // 仍然是"abc";賦值是無效的。沒有任何辦法修改字符串的內(nèi)容
而對象就不一樣了,對象是可變的。
var obj = {x : 0}; obj.x = 100; var o = obj; o.x = 1; obj.x; // 1, 被修改 o = true; obj.x; // 1, 不會(huì)因o = true改變
這里定義變量obj,值是object,然后設(shè)置obj.x屬性的值為100。而后定義另一個(gè)變量o,值仍然是這個(gè)object對象,此時(shí)obj和o兩個(gè)變量的值指向同一個(gè)對象(共享同一個(gè)對象的引用)。所以修改對象的內(nèi)容,對obj和o都有影響。但對象并非按引用傳遞,通過o = true修改了o的值,不會(huì)影響obj。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/79760.html
摘要:中所有函數(shù)的參數(shù)都是按值傳遞的?;绢愋捅旧硎前粗祩鬟f,具有不可變性,對基本類型的修改,實(shí)質(zhì)上都是在棧內(nèi)存中創(chuàng)建了新的值。中把這種拷貝也認(rèn)為是按值傳遞。 本文共 1200 字,讀完只需 5 分鐘 概述 參數(shù)的傳遞分為按值傳遞和按引用傳遞,而 JavaScript 中參數(shù)的傳遞只有按值傳遞。 ECMAScript 中所有函數(shù)的參數(shù)都是按值傳遞的。 所謂按值傳遞就是: 把函數(shù)外部的值復(fù)制給...
摘要:關(guān)于的求值策略,問中函數(shù)的參數(shù)傳遞是按值傳遞還是按引用傳遞回答很經(jīng)典。所以不能說中函數(shù)的參數(shù)傳遞嚴(yán)格按值傳遞或按引入傳遞。中還采用一種參數(shù)傳遞策略,叫按共享傳遞。中參數(shù)是必須先求值再作為實(shí)參傳入函數(shù)的。參考求值策略中函數(shù)參數(shù)的默認(rèn)值 最近在研究 lambda 演算中的 η-變換 在 JavaScript 中的應(yīng)用,偶然在 stackoverflow 上看到一個(gè)比較有意思的問題。關(guān)于 J...
這幾天遇到j(luò)s參數(shù)傳遞方式的問題,深切探究一番,將所得結(jié)果總結(jié)于此 常見的幾種傳遞方式 傳值調(diào)用(call by value) 在傳值調(diào)用中實(shí)際參數(shù)被求值,其值被綁定到函數(shù)中對應(yīng)的變量上(通常是把值復(fù)制到新內(nèi)存區(qū)域)。在函數(shù)返回后調(diào)用者作用域里的曾傳給函數(shù)的任何東西都不會(huì)變。 傳引用調(diào)用(call by reference) 在傳引用調(diào)用求值中,傳遞給函數(shù)的是它的實(shí)際參數(shù)的隱式引用(即實(shí)際參數(shù)的地...
摘要:最早由在年的語言中提出。該求值策略被用于等多種語言。該策略的重點(diǎn)是調(diào)用函數(shù)傳參時(shí),函數(shù)接受對象實(shí)參引用的副本既不是按值傳遞的對象副本,也不是按引用傳遞的隱式引用。它和按引用傳遞的不同在于在共享傳遞中對函數(shù)形參的賦值,不會(huì)影響實(shí)參的值。 周五晚上下班回家的路上,突然想到了 CommonJS 規(guī)范、Node.js 模塊化等等各種東西,然后就想到了熟悉的 module.exports。 大約...
摘要:熟悉的同學(xué)都知道,它的數(shù)據(jù)類型一般分為基本數(shù)據(jù)類型和引用數(shù)據(jù)類型復(fù)雜數(shù)據(jù)類型?;緮?shù)據(jù)類型基本數(shù)據(jù)類型有種新增。動(dòng)態(tài)的屬性基本數(shù)據(jù)類型的值是沒有辦法添加屬性和方法的。類型檢測根據(jù)它的原型鏈來識(shí)別引用類型。 熟悉js的同學(xué)都知道,它的數(shù)據(jù)類型一般分為基本數(shù)據(jù)類型和引用數(shù)據(jù)類型(復(fù)雜數(shù)據(jù)類型)。那么他們之間到底有什么不同?從下面這個(gè)例子入手: var a = 1; ...
閱讀 1878·2023-04-26 02:46
閱讀 2013·2021-11-25 09:43
閱讀 1153·2021-09-29 09:35
閱讀 2108·2019-08-30 15:56
閱讀 3432·2019-08-30 15:54
閱讀 2645·2019-08-29 16:35
閱讀 3130·2019-08-29 15:25
閱讀 3302·2019-08-29 14:01