摘要:在中其實是一顆語法糖,但是這糖有毒。致命的地方在于它的指向往往不能直觀確定。希望下面可以一步步去掉有毒的糖衣。這樣理解可能有些極端,但是它可能有助于避免一些常見的錯誤。第三步一個傳遞參數更好的辦法仍存在兩個安全隱患。
在 JavaScript 中 this 其實是一顆語法糖,但是這糖有毒。this 致命的地方在于它的指向往往不能直觀確定。希望下面可以一步步去掉有毒的糖衣。
1 用 f.call(thisVal, ...args) 指定 this調用函數的方式有三種,用 Function.prototype.call 調用可以指定 this:
定義 function f(...args){/*...*/}
調用 f.call(thisVal, ...args);
例一
function greet(){ console.log("Hello, " + this); } // 手動指定 `greet` 中的 `this`: greet.call("ngolin"); // Hello, ngolin
例二
function whoAreYou(){ console.log("I"m " + this.name); } whoAreYou.call({name: "Jane"}); // I"m Jane2 使用語法糖,this 自動指定
先接受函數 f 的正確調用方式是 f.call(thisVal, ...args);, 然后就可以把 f(...args); 理解成語法糖。
但是不用 f.call(thisVal, ...args), this 怎樣動態指定?
一、函數(function)
// 1. 在非嚴格模式下:window f(); // 解糖為 f.call(window); // 2. 但在嚴格模式下:undefined f(1, 2, 3); // 解糖為 f.call(undefined, 1, 2, 3);
一、方法(method)
// 無論是在嚴格還是非嚴格模式: obj.m(1, 2, 3); // 解糖為 obj.m.call(obj, 1, 2, 3); obj1.obj2.m(...args); // obj1.obj2.m.call(obj1.obj2, ...args); obj1.obj2....objn.m(); // obj1.obj2....objn.m.call(obj1.obj2....objn);
通過上面的例子,分別演示了函數 f(..args) 和方法 obj1.obj2....objn.m(..args) 怎樣自動指定 this.
嚴格區分函數(function)和方法(method)這兩個概念有利于清晰思考,因為它們在綁定 this 時發生的行為完全不一樣。同時函數和方法可以相互賦值(轉換),在賦值前后,唯一發生變化的是綁定 this 的行為(當然這種變化在調用時才會體現)。下面先看函數轉方法,再看方法轉函數。
3 函數轉方法函數聲明(function f(){})和函數表達式(var f = function(){};)有一些微妙的區別,但是兩種方式在調用時綁定this行為完全一樣,下面在嚴格模式下以函數表達式為例:
var f = function(){ console.log(this.name); }; var obj1 = { name: "obj 1", getName: f; }; var obj2 = { name: "obj 2", getName: f; }; // 函數 `f` 轉方法 `obj1.getName` obj1.getName();// "obj 1" => obj1.getName.call(obj1) // 不認為函數轉方法 obj2.getName.call(obj1);// "obj 1"(不是 "obj 2")
將函數轉成方法通常不太容易出錯,因為起碼在方法中 this 能夠有效地指向一個對象。函數轉成方法是一個模糊的說法,實際上可以這樣理解:
JavaScript 不能定義一個函數,也不能定義一個方法,是函數還是方法,要等到它執行才能確定;當把它當成函數執行,它就是函數,當把它當成方法執行,它就是方法。所以只能說執行一個函數和執行一個方法。
這樣理解可能有些極端,但是它可能有助于避免一些常見的錯誤。因為關系到 this 怎樣綁定,重要的是在哪里調用(比如在 obj1, obj2... 上調用)以及怎樣調用(比如以 f(), f.call()... 的方式),而不是在哪里定義。
但是,為了表達的方便,這里仍然會使用定義函數和定義方法這兩種說法。
4 方法轉函數將方法轉成函數比較容易出錯,比如:
var obj = { name: "obj", show: function(){ console.log(this.name); } }; var _show = obj.show; _show(); // error!! => _show.call(undefined) button.onClick = obj.show; button.onClick(); // error!! => button.onClick.call(button) (function(cb){ cb(); // error!! =>cb.call(undefined) })(obj.show);
當一個對象的方法使用了 this 時,如果這個方法最后不是由這個對象調用(比如由其他框架調用),這個方法就可能會出錯。但是有一種技術可以將一個方法(或函數)綁定(bind)在一個對象上,從而無論怎樣調用,它都能夠正常執行。
5 把方法綁定(bind)在對象上先看這個obj.getName的例子:
var obj = { getName: function(){ return "ngolin"; } }; obj.getName(); // "ngolin" obj.getName.call(undefined); // "ngolin" obj.getName.call({name: "ngolin"}); // "ngolin" var f = obj.getName; f(); // "ngolin" (function(cb){ cb(); // "ngolin" })(obj.getName);
上面的例子之所以可以成功是因為 obj.getName 根本沒有用到 this, 所以 this 指向什么對 obj.getName 都沒有影響。
這里有一種技術把使用 this 的方法轉成不使用 this 的方法,就是創建兩個閉包(即函數),第一個閉包將方法(method)和對象(obj)捕獲下來并返回第二個閉包,而第二個閉包用于調用并返回 obj.method.call(obj);. 下面一步步實現這種技術:
第一步 最簡單的情況下:
function method(){ obj.method.call(obj); } method(); // correct, :))
存在的缺陷:
只適合沒有參數和返回的 obj.method
存在兩個安全隱患:
1 后續改變 obj.method,比如 obj.method = null;
2 后續改變 obj,比如 obj = null
第二步 在方法有參數有返回的情況下:
function method(a, b){ return obj.method.call(obj, a, b); } method(a, b); // correct, :))
存在的缺陷:
只適合兩個參數的 obj.method
存在兩個安全隱患,同上。
第三步 一個傳遞參數更好的辦法:
function method(){ return obj.method.apply(obj, arguments); } method(a, b); // correct, :))
仍存在兩個安全隱患。
第四步 更加安全的方式:
var method = (function(){ return function(){ return obj.method.apply(obj, arguments); }; })(obj.method, obj); method(a, b); // correct, :))
第五步 抽象出一個函數,用于將方法綁定到對象上:
function bind(method, obj){ return function(){ return method.apply(obj, arguments); }; } var obj = { name: "ngolin", getName: function(){ return this.name; } }; var method = bind(obj.getName, obj); method(); // "ngolin"6 Function.prototype.bind
這種方法很常見,后來 ECMAScript 5 就增加了 Function.prototype.bind, 比如:
var binded = function(){ return this.name; }.bind({name: "ngolin"}); binded(); // "ngolin"
具體來說,Function.prototype.bind 這樣工作:
var bindedMethod = obj.method.bind(obj); // 相當于: var bindedMethod = (function(){ return function(){ return obj.method.apply(obj, arguments); }; })(obj.method, obj);
更多使用 Function.prototype.bind 的例子:
var f = obj.method.bind(obj); button.onClick = obj.method.bind(obj); document.addEventListener("click", obj.method.bind(obj));7 常見問題及容易出錯的地方
一 在定義對象時有沒有 this?
obj = { firstName: "First", lastName: "Last", // `fullName` 可以得到預期結果嗎? fullName: this.firstName + this.lastName } // 或者: function makePoint(article){ if(article.length <= 144) return article; return article.substr(0, 141) + "..."; } obj = { fulltext: "...a long article go here...", // `abstract` 呢? abstract: makePoint(this.fulltext) }
二 在方法內的 this 都是同一對象嗎?
obj = { count: 3, field: "field", method: function(){ function repeat(){ if(this.count > 100){ return this.field.repeat(this.count % 100); } this.field.repeat(this.count); }.bind(this); // 這個呢? return repeat(); } }
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/89453.html
摘要:輸出的作用與和一樣,都是可以改變函數運行時上下文,區別是和在調用函數之后會立即執行,而方法調用并改變函數運行時上下文后,返回一個新的函數,供我們需要時再調用。 前言 js中的call(), apply()和bind()是Function.prototype下的方法,都是用于改變函數運行時上下文,最終的返回值是你調用的方法的返回值,若該方法沒有返回值,則返回undefined。這幾個方法...
摘要:在和中都保留了數組的強引用,所以在中簡單的清除變量內存并沒有得到釋放,因為還存在引用計數。而在中,它的鍵是弱引用,不計入引用計數中,所以當被清除之后,數組會因為引用計數為而被回收掉。其實我們主要注意的引用是不計引用計數的,就好理解了。 showImg(https://segmentfault.com/img/remote/1460000019147368?w=900&h=383); 前...
摘要:中基本都使用來開發,但其實是的一種語法糖。但是我們必須知道,本質上就是在編譯的時候,會由將轉化為。比如生成了比如生成了解的本質,只需要記住本質就是附錄提供的一個在線轉換為的地址 react中基本都使用JSX來開發,但JSX其實是javascript的一種語法糖。 什么是語法糖? 語法糖就是提供了一種全新的方式書寫代碼,但是其實現原理與之前的寫法相同。語法糖可以說是廣泛存在于各種計算機...
摘要:題目要求假設有個孩子站成一排,每個孩子擁有一個評估值。我們可以觀察到,每次最遠只需要額外分發到距離當前最近的評分最高的那個孩子。因為他的糖果數量的增加并不會影響到之前孩子。當有多個最近評分最高的孩子時,則選擇最后一個。 題目要求 There are N children standing in a line. Each child is assigned a rating value....
摘要:貪心法復雜度時間空間思路典型的貪心法,如果一個孩子比另一個孩子的分高,我們只多給塊糖。我們可以先從左往右遍歷,確保每個孩子根他左邊的孩子相比,如果分高,則糖要多個,如果分比左邊低,就只給一顆。 Candy There are N children standing in a line. Each child is assigned a rating value. You are gi...
閱讀 3049·2021-09-22 15:52
閱讀 2914·2019-08-30 15:55
閱讀 2708·2019-08-30 15:53
閱讀 2461·2019-08-30 13:21
閱讀 1630·2019-08-30 13:10
閱讀 2488·2019-08-26 12:09
閱讀 2575·2019-08-26 10:33
閱讀 1810·2019-08-23 18:06