最近遇到個有趣的問題:「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。

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

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

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

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

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

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

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

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


熱AI工具

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

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

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

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

熱門文章

熱工具

WebStorm Mac版
好用的JavaScript開發工具

SublimeText3 Linux新版
SublimeText3 Linux最新版

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

SublimeText3漢化版
中文版,非常好用

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