JavaScript中的this引用

来源:http://www.cnblogs.com/thislbq/archive/2016/08/04/5738811.html
-Advertisement-
Play Games

在JavaScript的學習當中,this關鍵字的出現頻率可不低,所以想想有必要對this關鍵字做一個總結。在總結過程中,參考的資料來源於書本及網上。 一、定義 1、this是函數內部的一個特殊對象(或this引用)--它引用的是函數據以執行的環境對象。(來源於JavaScript高級程式設計) 2 ...


JavaScript的學習當中,this關鍵字的出現頻率可不低,所以想想有必要對this關鍵字做一個總結。在總結過程中,參考的資料來源於書本及網上。

 

一、定義

1、this是函數內部的一個特殊對象(或this引用)--它引用的是函數據以執行的環境對象。(來源於JavaScript高級程式設計)

2、this引用是一種在JavaScript的代碼中隨時都可以使用的只讀變數。 this引用 引用(指向)的是一個對象,它有著會根據代碼上下文語境自動改變其引用對象的特性。它的引用規則如下:

• 在最外層代碼中,this引用 引用的是全局對象。

• 在函數內,this引用根據函數調用的方式的不同而有所不同。如下

1)構造函數的調用--this引用 引用的是所生成的對象

2)方法調用--this引用 引用的是接收方對象

3)apply或call調用--this引用 引用的是有apply或call的參數指定的對象

4)其他方式的調用--this引用 引用的是全局對象

(來源於JavaScript編程全解)

 

二、根據以上所述及網上的相關資料,this對象(引用)的使用情況總結如下:

JavaScript是動態語言,this關鍵字在執行的時候才能確定是誰。所以this永遠指向調用者,即對“調用對象”的引用。簡單點說就是調用的方法屬於哪個對象,this就指向那個對象。根據函數調用方式的不同,this可以 指向全局對象,當前對象,或其他任意對象。

 

  1、全局函數調用,全局函數中的this會指向全局對象window。(函數調用模式)

 1 //代碼清單1
 2 <script type="text/javascript">
 3     var message = "this in window";    //這一句寫在函數外面和裡面是一樣效果
 4     function func() {
 5         if(this == window){
 6             alert("this == window"); 
 7             alert(message);
 8             this.methodA = function() {
 9                 alert("I'm a function");
10                 }
11         }
12     }
13 
14     func();   //如果不調用func方法,則裡面定義的屬性或方法會取不到   
15     methodA();
16 </script>

func()的調用結果為this == window, this in window

methodA()的調用結果為I'm a function

 

  2、構造函數調用,即使用new的方式實例化一個對象,this會指向通過構造函數生成的對象。(構造器調用模式)

 1 代碼清單2
 2 <script type="text/javascript">
 3     function Func() {
 4         if (this == window) {
 5             alert("this == window");
 6         }
 7         else {
 8             alert("this != window");
 9         }
10         this.fieldA = "I'm a field";
11         alert(this);
12     }
13 
14     var obj = new Func();
15     alert(obj.fieldA);    //this指向的是對象obj
16 </script>

 

  3、對象方法的調用,this指向當前對象。任何函數,只要該函數被當做一個對象的方法使用或賦值時,該函數內部的this都是對該對象本身的引用。也可理解為this寫在一個普通對象中,this向的就是對象本身。(方法調用模式)

(方法的定義: 作為對象屬性的函數稱為方法)

 1 //代碼清單3
 2 <script type="text/javascript">
 3     var obj = {
 4         x: 3,
 5         doit: function(){
 6             if(this == window){
 7                 alert("this == window");
 8             }else{
 9                 alert("method is called: " + this.x);
10             }
11         }
12     };
13     
14     obj.doit();    //this指向的是對象obj
15 </script>

 

  4、通過apply或call方法調用,this指向傳入的對象。

apply 或call 方法可以用來代替另一個對象調用一個方法。call 方法可將一個函數的對象上下文從初始的上下文改變為由 thisObj 指定的新對象。如果沒有提供 thisObj 參數,那麼 Global 對象被用作 thisObj。  (apply調用模式)

 1 //代碼清單4
 2 <script type="text/javascript">
 3     var obj = {
 4         x: 3,
 5         doit: function(){
 6             alert("method is called: " + this.x);
 7         }
 8     };
 9     var obj2 = {x: 4};
10     
11     obj.doit();    //3,this指向obj
12     obj.doit.apply(obj2);    //4,this指向obj2
13     obj.doit.call(obj2);    //4,this指向obj2
14 </script>

 

  5、原型鏈中的this --原型對象及構造函數中的this指向新創建的實例對象。使用prototype擴展方法可以使用this獲取到源對象的實例,私有欄位無法通過原型鏈獲取

 1 //代碼清單5
 2 <script type="text/javascript">
 3     function Func() {
 4         this.fieldA = "I'm a field";
 5         var privateFieldA = "I'm a var";
 6     }
 7 
 8     Func.prototype = {
 9         ExtendMethod: function(str) {
10             alert(str + " :" + this.fieldA);
11             alert(privateFieldA);  //出錯,私有欄位無法通過原型鏈獲取。
12         }
13      };
14      
15     var obj = new Func();
16     obj.ExtendMethod("From prototype");   //此時構造函數及原型鏈中的this指向對象obj
17 </script>

 

  6、閉包中的this --閉包:寫在function中的function,this指向全局對象window

   6.1 對象中的閉包

 1 //代碼清單6
 2 <script type="text/javascript">
 3     var name = "The window";
 4     var obj = {
 5         name: "My Object",
 6         getNameFunc: function(){
 7             return function(){
 8                 return this.name;
 9             }
10         }
11     };
12     
13     alert(obj.getNameFunc()());    //The window
14 </script>

此時,閉包中的this指向全局對象window,只能取到全局對象的屬性。那麼對象內部的屬性(外部函數的變數)要想訪問又怎麼辦呢? 把外部函數的this對象保存在一個閉包能訪問的變數就可以了。看如下代碼:

 1 //代碼清單7
 2 <script type="text/javascript">
 3     var name = "The window";
 4     var obj = {
 5         name: "My Object",
 6         getNameFunc: function(){
 7             var that = this;
 8             return function(){
 9                 return that.name;
10             }
11         }
12     };
13     
14     alert(obj.getNameFunc()());    //My object
15 </script>

將外部函數的this賦值給that變數,就能讀取到外部函數的變數。

   6.2 不管是直接引用function,還是實例化一個function,其返回的閉包函數里的this都是指向window

 1 //代碼清單8
 2 <script type="text/javascript">
 3     function a() {
 4         alert(this == window);
 5         var that = this;
 6         var func = function() {
 7             alert(this == window);
 8             alert(that);
 9         };
10         return func;
11      }
12 
13     var b = a();
14     b();   //true, true, [object Window]
15     var c = new a();
16     c();  //false, true, [object object]
17 </script>

 

  7、函數使用bind()方法綁定一個對象,this會指向傳給bind()函數的值。

 1 //代碼清單9
 2 <script type="text/javascript">
 3     window.color = "red";
 4     var obj = {color: "blue"};
 5     function sayColor(){
 6         alert(this.color);
 7     }
 8     
 9     var objSayColor = sayColor.bind(obj);
10     objSayColor();   //blue
11 </script>

 

  8、內嵌在HTML元素中的腳本段,this指向元素本身

1 //代碼清單10
2 <div onclick="test(this)" id="div">Click Me</div>
3 <script type="text/javascript">
4     function test(obj) {
5     alert(obj);   //[object HTMLDivElement]
6     }
7 </script>

 

  9寫在script標簽中:this就是指全局對象window。這個跟第一點的全局函數調用的全局變數一樣。

以上總結的情況未必完整,若在工作中發現有其他情況再補充進來。

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文為原創文章,轉載請註明出處,謝謝 數據的發佈與訂閱 1、應用 服務端監聽數據改變,客戶端創建/更新節點數據,客戶端提供數據,服務端處理 2、原理 客戶端監控節點數據改變事件(例如配置信息,下圖的config節點),啟動時在伺服器節點下創建臨時節點(圖中servers下節點) 服務端監聽工作伺服器 ...
  • 1.定義 定義對象間一種一對多的依賴關係,使得當每一個對象改變狀態,則所有依賴於它的對象都會得到通知並自動更新。 2.類圖 3.代碼示例 我們定義一個場景:熱水壺在燒開水,小孩和媽媽都關註燒開水的過程,各自有其處理方法。用while死迴圈一直輪詢雖然可以實現這樣的場景,但性能上讓人無法接受。 為方便 ...
  • ajax簡介 AJAX即“Asynchronous Javascript And XML”(非同步JavaScript和XML),是指一種創建互動式網頁應用的網頁開發技術。Ajax不是一種新的編程語言,而是使用現有標準的新方法。AJAX可以在不重新載入整個頁面的情況下,與伺服器交換數據。這種非同步交互的 ...
  • soChange一款多很經典的幻燈片的jQuery插件。 實例預覽 引入文件 複製 使用方法 複製 複製 soChange參數 複製 soChange 即 simple object change ,對象切換插件,充分發揮css樣式的靈活性,不僅僅適用於圖片相冊,也適用於選項卡或文字類型的切換,以上 ...
  • js代碼: 游戲的對象 ,食物,蛇 ,游戲控制思路如下 (完整代碼在https://github.com/774044859yf/ObjectSnakeGame下載) var snake = { ...
  • [1]參數預設值 [2]rest參數 [3]擴展運算符 [4]箭頭函數 ...
  • jQuery.data的是jQuery的數據緩存系統。它的主要作用就是為普通對象或者DOM元素添加數據。 1 內部存儲原理 這個原理很簡單,原本要添加在DOM元素本身的數據,現在被集中的存儲在cache集合中。它們之間靠一個從1開始的數字鍵來聯繫著。這樣DOM元素就不會像以前那麼笨重了,更不會出現以... ...
  • 一、引言 前面我們介紹了有關於內置對象的很多很多的API,講道理得話如果想徹底的掌握那一定要經過一定的代碼段沉澱下。大家可以想象一下,既然在程式中有很多的內置對象供我們使用,那我們是不是也可以定義一些對象作為我們自己的特用對象呢?答案肯定是可以的。 二、導入 在工作中根據內容需求不同我們可以定義不同 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...