首頁 web前端 js教程 JavaScript中的函數呼叫的程式碼詳情

JavaScript中的函數呼叫的程式碼詳情

Mar 23, 2017 pm 02:26 PM

可能很多人在學習JavaScript 過程中碰到過函數參數傳遞方式的迷惑,本著深入的精神,我給大家分享了一篇教程關於javascript中的函數呼叫知識,有興趣的朋友一起學習吧

定義

可能很多人在學習Javascript 過程中碰到過函數參數傳遞方式的迷惑,本著深入的精神,我想再源碼中尋找些答案不過在做這件事之前,首先明確幾個概念。拋棄掉值傳遞、引用傳遞等固有叫法,回歸英文:

call by reference && call by value && call by sharing

#分別是我們理解的C++ 中的引用傳遞,值傳遞。第三種比較迷惑,官方解釋是 receives the copy of the reference to object 。我用通俗的話解釋一下:

Object 可以理解為key 的集合,Object 對key 指向的資料是引用性質的(這裡不深究是指標實作還是C++引用實作),函數接收的是一個變數的copy,變數包含了Object 的引用,是一個值傳遞。

那麼很明顯,函數傳參的時候我們接收到的物件型參其實是實參的複製,所以直接更改型參的指向是不可行的;由於Object 本身的key 都是引用,所以修改key 的指向是可行的。

證明

簡單來幾段程式碼即可證明

Code 1: 函式能修改key 所指向的資料

let func = obj => { obj.name = 'Dosk' };
let obj = {name : 'Alxw'};
console.log(obj); //{ name: 'Alxw' }
func(obj)
console.log(obj); //{ name: 'Dosk' }
登入後複製

Code 2: 函數不能修改obj

let func = obj => { obj = {} };
let obj = {name : 'Alxw'};
console.log(obj); //{ name: 'Alxw' }
func(obj)
console.log(obj); //{ name: 'Alxw' }
登入後複製

Code 3: 內部obj 和外部=== 結果相等

let def = {name : 'Alxw'};
let func = obj => { console.log(obj === def) };
func(def); //true
登入後複製

所以第三段程式碼可能有疑問了,既然obj 是def 的複製,為什麼=== 操作還能夠為真?不是說 === 操作對於 Object 比較的是在記憶體中的位址麼,如果複製應該是 false 才對啊?

所以我們回到 Google V8 的原始碼來看這件事。

深入Google V8

我們來看看原始碼嚴格等於操作碼部分:

bool Object::StrictEquals(Object* that) {
 if (this->IsNumber()) {
  if (!that->IsNumber()) return false;
  return NumberEquals(this, that);
 } else if (this->IsString()) {
  if (!that->IsString()) return false;
  return String::cast(this)->Equals(String::cast(that));
 } else if (this->IsSimd128Value()) {
  if (!that->IsSimd128Value()) return false;
  return Simd128Value::cast(this)->Equals(Simd128Value::cast(that));
 }
 return this == that;
}
登入後複製

看起來應該是最後一種情況,理論上如果def 和obj 是不同的對象,那麼應該返回false 才對,這不是推翻了上文所述麼?其實不,忽略了一件事,即Google V8 內部在實例化一個Object 的時候,本身就是動態實例化,而我們知道在編譯型語言中如果動態實例化只能夠在堆內存上,即只能夠指針引用。這個結論是的證明涉及到Local 、Handle 等class 的實現,我覺得太麻煩,有一個簡單的證明方式,即搜尋源碼得到所有呼叫Object::StrictEquals 的地方都是直接傳入而沒有取位址運算。

不過有人會問,既然是值傳遞的變數包含 Object 的引用,理論上也能夠修改 Object 才對,為什麼第三段程式碼不能修改呢?

很簡單的道理,因為我們在Javascript 語言邏輯層次上的所謂的操作,只不過是在調用Google V8 的實例方的法而已,根本不可能操作到這一地步(當然,潛在的BUG 不算的-。

##的確,傳遞的時候是值傳遞,但是內容包含了Object 的指針,而且不能夠修改這個指針,他是多個變數共享的。

另一種簡單的證明

來來來,看原始碼

V8_DEPRECATE_SOON("Use maybe version",
         Local<Value> Call(Local<Value> recv, int argc,
                  Local<Value> argv[]));
V8_WARN_UNUSED_RESULT MaybeLocal<Value> Call(Local<Context> context,
                       Local<Value> recv, int argc,
                       Local<Value> argv[]);
登入後複製
上面的是即將棄用的

介面,碰巧我看到的這個版本程式碼包含大量的這種即將棄用的程式碼,看看就好。重點是第二個接口,是函數的唯一的調用的接口。裡面的 Local 最後會呼叫 C++ 的位元複製,所以可以簡單的證明就是值傳遞。

可能是重點

別忘了,我們定義的變數都是類似Handle

這種形式的,所以它們之間物件才是共享的,我們所說的Javascript 裡面變數不直接指的是Object 的實例!!!

最後的最後

總之理解起來可能很費力甚至有錯誤,但是在 Javascript 語言層次上能夠確定了特性,這才是重要的。

以上所述是小編給大家介紹的Javascript中的函數調用,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回覆大家的。在此也非常感謝大家對腳本之家網站的支持!

以上是JavaScript中的函數呼叫的程式碼詳情的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
4 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

如何使用WebSocket和JavaScript實現線上語音辨識系統 如何使用WebSocket和JavaScript實現線上語音辨識系統 Dec 17, 2023 pm 02:54 PM

如何使用WebSocket和JavaScript實現線上語音辨識系統引言:隨著科技的不斷發展,語音辨識技術已成為了人工智慧領域的重要組成部分。而基於WebSocket和JavaScript實現的線上語音辨識系統,具備了低延遲、即時性和跨平台的特點,成為了廣泛應用的解決方案。本文將介紹如何使用WebSocket和JavaScript來實現線上語音辨識系

WebSocket與JavaScript:實現即時監控系統的關鍵技術 WebSocket與JavaScript:實現即時監控系統的關鍵技術 Dec 17, 2023 pm 05:30 PM

WebSocket與JavaScript:實現即時監控系統的關鍵技術引言:隨著互聯網技術的快速發展,即時監控系統在各個領域中得到了廣泛的應用。而實現即時監控的關鍵技術之一就是WebSocket與JavaScript的結合使用。本文將介紹WebSocket與JavaScript在即時監控系統中的應用,並給出程式碼範例,詳細解釋其實作原理。一、WebSocket技

如何利用JavaScript和WebSocket實現即時線上點餐系統 如何利用JavaScript和WebSocket實現即時線上點餐系統 Dec 17, 2023 pm 12:09 PM

如何利用JavaScript和WebSocket實現即時線上點餐系統介紹:隨著網路的普及和技術的進步,越來越多的餐廳開始提供線上點餐服務。為了實現即時線上點餐系統,我們可以利用JavaScript和WebSocket技術。 WebSocket是一種基於TCP協定的全雙工通訊協議,可實現客戶端與伺服器的即時雙向通訊。在即時線上點餐系統中,當使用者選擇菜餚並下訂單

C++ 函式呼叫效能調優:參數傳遞與傳回值的影響 C++ 函式呼叫效能調優:參數傳遞與傳回值的影響 May 04, 2024 pm 12:57 PM

C++函式呼叫效能最佳化包含兩個面向:參數傳遞策略和傳回值類型最佳化。參數傳遞方面,傳遞值適合小物件、不可修改參數,傳遞參考或指標則適合大物件、可修改參數,而傳遞指標速度最快。傳回值最佳化方面,小型值可直接傳回,大物件應傳回引用或指標。選擇合適策略能提高函數呼叫效能。

JavaScript與WebSocket:打造高效率的即時天氣預報系統 JavaScript與WebSocket:打造高效率的即時天氣預報系統 Dec 17, 2023 pm 05:13 PM

JavaScript和WebSocket:打造高效的即時天氣預報系統引言:如今,天氣預報的準確性對於日常生活以及決策制定具有重要意義。隨著技術的發展,我們可以透過即時獲取天氣數據來提供更準確可靠的天氣預報。在本文中,我們將學習如何使用JavaScript和WebSocket技術,來建立一個高效的即時天氣預報系統。本文將透過具體的程式碼範例來展示實現的過程。 We

C++ 中如何在不同模組中呼叫函式? C++ 中如何在不同模組中呼叫函式? Apr 12, 2024 pm 03:54 PM

在C++中跨模組呼叫函數:宣告函數:在目標模組的頭檔中宣告要呼叫的函數。實作函數:在原始檔中實作函數。連結模組:使用連結器將包含函數宣告和實作的模組連結在一起。呼叫函數:在需要呼叫的模組中包含目標模組的頭文件,然後呼叫函數。

簡易JavaScript教學:取得HTTP狀態碼的方法 簡易JavaScript教學:取得HTTP狀態碼的方法 Jan 05, 2024 pm 06:08 PM

JavaScript教學:如何取得HTTP狀態碼,需要具體程式碼範例前言:在Web開發中,經常會涉及到與伺服器進行資料互動的場景。在與伺服器進行通訊時,我們經常需要取得傳回的HTTP狀態碼來判斷操作是否成功,並根據不同的狀態碼來進行對應的處理。本篇文章將教你如何使用JavaScript來取得HTTP狀態碼,並提供一些實用的程式碼範例。使用XMLHttpRequest

C++ 函數呼叫反射技術:參數傳遞和傳回值的動態訪問 C++ 函數呼叫反射技術:參數傳遞和傳回值的動態訪問 May 05, 2024 am 09:48 AM

C++函數呼叫反射技術允許在運行時動態獲取函數的參數和返回值資訊。使用typeid(decltype(...))和decltype(...)表達式可取得參數和傳回值類型資訊。透過反射,我們可以動態呼叫函數,並根據運行時輸入選擇特定的函數,實現靈活且可擴展的程式碼。

See all articles