搜尋
首頁web前端js教程JavaScript中的值是按值傳遞還是按引用傳遞問題探討_javascript技巧

最近遇到個有趣的問題:「JS中的值是按值傳遞,還是按引用傳遞呢?」
 
在分析這個問題之前,我們需要先了解什麼是按值傳遞(call by value),什麼是按引用傳遞(call by reference)。在計算機科學裡,這個部分叫做求值策略(Evaluation Strategy)。它決定變數之間、函數呼叫時實參和形參之間值是如何傳遞的。
 
以數值傳遞 VS. 依引用傳遞

以值傳遞(call by value)是最常用的求值策略:函數的形參是被呼叫時所傳實參的副本。修改形參的值並不會影響實參。
 
依引用傳遞(call by reference)時,函數的形參接收實參的隱式引用,而不再是副本。這意味著函數形參的值如果被修改,實參也會被修改。同時兩者指向相同的值。
 
按引用傳遞會使函數呼叫的追蹤更加困難,有時也會引起一些微妙的BUG。
 
按值傳遞由於每次都需要克隆副本,對一些複雜類型,性能較低。兩種傳值方式都有各自的問題。
 
我們先看一個C的例子來了解按值和引用傳遞的區別:

複製程式碼 程式碼如下:

void Modify(int p, int * q)
{
    p = 27; // 依數值傳遞 – p是實參a的複製, 只有p被修改
    *q = 27; // q是b的引用,q和b都被修改
}
int main()
{
    int a = 1;
    int b = 1;
    Modify(a, &b);   // a 按值傳遞, b 依參考傳遞,
                     // a 未改變, b 改變了
    return(0);
}

這裡我們可以看到:
 
a => p按值傳遞時,修改形參p的值並不影響實參a,p只是a的副本。
b => q是依照引用傳遞,修改形參q的值時也影響到了實參b的值。
探究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是同一個對象,o不是obj的副本。所以不是按值傳遞。 但這樣是否說明JS的物件是按引用傳遞的呢?我們再看下面的例子:
複製程式碼 程式碼如下:

var obj = {x : 1};
function foo(o) {
    o = 100;
}
foo(obj);
console.log(obj.x); // 仍然是1, obj並未被修改為100.

如果是按引用傳遞,修改形參o的值,應該影響到實參才對。但這裡修改o的值並未影響obj。 因此JS中的物件並不是按引用傳遞。那麼究竟物件的值在JS中如何傳遞的呢?
 
依共享傳遞 call by sharing

準確的說,JS中的基本類型按值傳遞,物件類型按共享傳遞的(call by sharing,也叫按物件傳遞、按物件共享傳遞)。最早由Barbara Liskov. 在1974年的GLU語言中提出。此求值策略被用於Python、Java、Ruby、JS等多種語言。
 
此策略的重點是:呼叫函數傳參時,函數接受物件實參引用的副本(既不是按值傳遞的物件副本,也不是按引用傳遞的隱式引用)。 它和依引用傳遞的不同在於:在共享傳遞中對函數形參的賦值,不會影響實參的值。如下面例子中,不可以透過修改形參o的值,來修改obj的值。

複製程式碼 程式碼如下:

var obj = {x : 1};
function foo(o) {
    o = 100;
}
foo(obj);
console.log(obj.x); // 仍然是1, obj並未被修改為100.

然而,雖然引用是副本,引用的物件是相同的。它們共用相同的對象,所以修改形參對象的屬性值,也會影響到實參的屬性值。
複製程式碼 程式碼如下:

var obj = {x : 1};
function foo(o) {
    o.x = 3;
}
foo(obj);
console.log(obj.x); // 3, 被修改了!

對於物件類型,由於物件是可變(mutable)的,修改物件本身會影響到共享這個物件的引用和引用副本。而對於基本類型,由於它們都是不可變的(immutable),按共享傳遞與按值傳遞(call by value)沒有任何區別,所以說JS基本類型既符合按值傳遞,也符合按共享傳遞。
複製程式碼 程式碼如下:

var a = 1; // 1是number型,不可變 var b = a; b = 6;

根據共享傳遞的求值策略,a和b是兩個不同的引用(b是a的引用副本),但引用相同的值。由於這裡的基本類型數字1不可變,所以這裡說按值傳遞、按共享傳遞沒有任何區別。
 
基本類型的不可變(immutable)性質

基本型別是不可變的(immutable),只有物件是可變的(mutable). 例如數字值100, 布林值true, false,修改這些值(例如把1變成3, 把true變成100 )並沒有什麼意義。比較容易誤解的,是JS中的string。有時我們會嘗試「改變」字串的內容,但在JS中,任何看似對string值的」修改」操作,實際上都是創建新的string值。

複製程式碼 程式碼如下:

var str = “abc”;
str[0]; // “a”
str[0] = “d”;
str; // 仍然是”abc”;賦值是無效的。沒有任何辦法修改字串的內容

而物件就不一樣了,物件是可變的。
複製程式碼 程式碼如下:

var obj = {x : 1};
obj.x = 100;
var o = obj;
o.x = 1;
obj.x; // 1, 被修改
o = true;
obj.x; // 1, 不會因o = true改變

這裡定義變數obj,值是object,然後設定obj.x屬性的值為100。而後定義另一個變數o,值仍然是這個object對象,此時obj和o兩個變數的值指向同一個對象(共享同一個對象的引用)。所以修改對象的內容,對obj和o都有影響。但物件並非按引用傳遞,透過o = true修改了o的值,不會影響obj。
陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
Python vs. JavaScript:開發人員的比較分析Python vs. JavaScript:開發人員的比較分析May 09, 2025 am 12:22 AM

Python和JavaScript的主要區別在於類型系統和應用場景。 1.Python使用動態類型,適合科學計算和數據分析。 2.JavaScript採用弱類型,廣泛用於前端和全棧開發。兩者在異步編程和性能優化上各有優勢,選擇時應根據項目需求決定。

Python vs. JavaScript:選擇合適的工具Python vs. JavaScript:選擇合適的工具May 08, 2025 am 12:10 AM

選擇Python還是JavaScript取決於項目類型:1)數據科學和自動化任務選擇Python;2)前端和全棧開發選擇JavaScript。 Python因其在數據處理和自動化方面的強大庫而備受青睞,而JavaScript則因其在網頁交互和全棧開發中的優勢而不可或缺。

Python和JavaScript:了解每個的優勢Python和JavaScript:了解每個的優勢May 06, 2025 am 12:15 AM

Python和JavaScript各有優勢,選擇取決於項目需求和個人偏好。 1.Python易學,語法簡潔,適用於數據科學和後端開發,但執行速度較慢。 2.JavaScript在前端開發中無處不在,異步編程能力強,Node.js使其適用於全棧開發,但語法可能複雜且易出錯。

JavaScript的核心:它是在C還是C上構建的?JavaScript的核心:它是在C還是C上構建的?May 05, 2025 am 12:07 AM

javascriptisnotbuiltoncorc; sanInterpretedlanguagethatrunsonenginesoftenwritteninc.1)JavascriptwasdesignedAsignedAsalightWeight,drackendedlanguageforwebbrowsers.2)Enginesevolvedfromsimpleterterpretpretpretpretpreterterpretpretpretpretpretpretpretpretpretcompilerers,典型地,替代品。

JavaScript應用程序:從前端到後端JavaScript應用程序:從前端到後端May 04, 2025 am 12:12 AM

JavaScript可用於前端和後端開發。前端通過DOM操作增強用戶體驗,後端通過Node.js處理服務器任務。 1.前端示例:改變網頁文本內容。 2.後端示例:創建Node.js服務器。

Python vs. JavaScript:您應該學到哪種語言?Python vs. JavaScript:您應該學到哪種語言?May 03, 2025 am 12:10 AM

選擇Python還是JavaScript應基於職業發展、學習曲線和生態系統:1)職業發展:Python適合數據科學和後端開發,JavaScript適合前端和全棧開發。 2)學習曲線:Python語法簡潔,適合初學者;JavaScript語法靈活。 3)生態系統:Python有豐富的科學計算庫,JavaScript有強大的前端框架。

JavaScript框架:為現代網絡開發提供動力JavaScript框架:為現代網絡開發提供動力May 02, 2025 am 12:04 AM

JavaScript框架的強大之處在於簡化開發、提升用戶體驗和應用性能。選擇框架時應考慮:1.項目規模和復雜度,2.團隊經驗,3.生態系統和社區支持。

JavaScript,C和瀏覽器之間的關係JavaScript,C和瀏覽器之間的關係May 01, 2025 am 12:06 AM

引言我知道你可能會覺得奇怪,JavaScript、C 和瀏覽器之間到底有什麼關係?它們之間看似毫無關聯,但實際上,它們在現代網絡開發中扮演著非常重要的角色。今天我們就來深入探討一下這三者之間的緊密聯繫。通過這篇文章,你將了解到JavaScript如何在瀏覽器中運行,C 在瀏覽器引擎中的作用,以及它們如何共同推動網頁的渲染和交互。 JavaScript與瀏覽器的關係我們都知道,JavaScript是前端開發的核心語言,它直接在瀏覽器中運行,讓網頁變得生動有趣。你是否曾經想過,為什麼JavaScr

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

WebStorm Mac版

WebStorm Mac版

好用的JavaScript開發工具

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

Safe Exam Browser

Safe Exam Browser

Safe Exam Browser是一個安全的瀏覽器環境,安全地進行線上考試。該軟體將任何電腦變成一個安全的工作站。它控制對任何實用工具的訪問,並防止學生使用未經授權的資源。