成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

javascript的this簡(jiǎn)單記憶

Hujiawei / 2872人閱讀

摘要:一直感覺中的是一個(gè)大坑,雖然一直都有注意,一直都有研究,但是總是會(huì)忘記。即使也沒有權(quán)利改變。但是可以影響后面的參數(shù)值。由于只是我根據(jù)經(jīng)驗(yàn)和資料自己總結(jié)的,所以不知道是否有疏忽或者遺漏,如果有問題的地方,歡迎提出。

一直感覺javascript中的this是一個(gè)大坑,雖然一直都有注意,一直都有研究,但是總是會(huì)忘記。在這里,總結(jié)一下一些簡(jiǎn)單的規(guī)律

不考慮箭頭函數(shù)的情況下

下面的所有的都是不考慮箭頭函數(shù)的情況下,因?yàn)榧^函數(shù)和普通的差別很大

直接調(diào)用的都是window

除非是bind綁定過的,其他情況下,直接調(diào)用的方法的this都是window。所謂的直接調(diào)用,就是直接以method()的形式調(diào)用,沒有call, apply, new

看幾種情況:

function foo(){
   return this;
}
foo(); // window
var obj = {
  foo: function(){ return this; }
}

var foo = obj.foo;
foo(); // window
function Foo(){
  this.foo = function(){
    console.log(this);
  }

  var foo = this.foo;
  foo(); // window
}

new Foo();
誰調(diào)用,誰是this

除非是bind綁定過的,一般情況下,誰調(diào)用這個(gè)方法,那么內(nèi)部的this就指向這個(gè)對(duì)象。也就是說obj.method(),那么就指向obj。obj.foo.method(),那么就指向obj.foo

看幾個(gè)例子:

var obj = {
  foo: function(){ return this; }
}
obj.foo(); // obj調(diào)用,所以結(jié)果是obj
function foo(){ return this };
var obj = {};
obj.foo = foo;
obj.foo(); // obj調(diào)用,所以結(jié)果是obj
var obj = {
  bar: function(){ return this; },
  foo: function(){
    return this.bar();
  }
}
obj.foo(); // 在foo中, this是obj, 而this調(diào)用的bar, 所以返回的是obj
var obj = {
   bar: {
     foo: function(){ return this }
   }
}
obj.bar.foo(); // obj.bar調(diào)用的foo,所以返回的結(jié)果是bar
function foo(){
  this.bar = function(){ return this }
  return this.bar();
}
foo(); // 由于foo中的this是window, 所以this.bar()返回的是window
function Foo(){
  this.foo = function(){ console.log(this); }
  this.foo();
}
var object = new Foo(); // 由于this.foo中的this是object,所以this是object
new會(huì)生成一個(gè)新的this

所有情況下,(箭頭函數(shù)不能使用new關(guān)鍵字),使用了new以后,會(huì)把內(nèi)部的this指向新生成的對(duì)象。

除去bind的情況下,prototype中的this也指向新生成的對(duì)象

function Foo(){
  console.log(this); // this指向新生成的對(duì)象,object
}
var object = new Foo(); 
function Foo(){
  this.foo = function(){ return this === object; }
}
var object = new Foo();
object.foo(); // 輸出true
function Foo(){}
Foo.prototype.foo = function(){ return this === object; }
var object = new Foo();
object.foo(); // 輸出true
call, apply是誰,this就是誰

除非是bind的情況,call, apply是誰,那么內(nèi)部的this就是誰。

注意:如果是基本類型,那么javascript會(huì)把基本類型轉(zhuǎn)換成Object的形式

也是看例子:

function foo(){ return this; }
foo.call("a"); // String
typeof foo.call("a"); // object
var obj = {
  foo : function(){ return this; }
}
obj.foo.call(1); // Number
typeof obj.foo.call(1); // object
function Foo(){
  this.foo = function(){ return this; }
}
var object = new Foo();
object.foo.call(1); // Number
bind是誰,this就是誰

除了new這一種特殊情況,bind的對(duì)象是誰,那么this也就是誰。即使call, apply也沒有權(quán)利改變。

注意:如果是基本類型,那么javascript會(huì)把基本類型轉(zhuǎn)換成Object的形式

function foo() { return this; } 
foo = foo.bind(1);
foo(); // Number
typeof foo(); // object
function foo() { return this; } 
foo = foo.bind(1);
foo.call("a"); // Number 1
function Foo() { return this; } 
Foo.prototype.foo = (function(){ return this; }).bind(1);
var object = new Foo();
object.foo(); // Number

特殊情況
new這個(gè)關(guān)鍵詞可以改變方法內(nèi)部的this,使他指向新生成的對(duì)象

function Foo(){ this.foo = function(){ console.log(this === obj) } }
Foo = Foo.bind(1);
var obj = new Foo();
obj.foo(); // 輸入true
箭頭函數(shù)

箭頭函數(shù)的"this"是根據(jù)定義環(huán)境的this來定的,也就是說定義的函數(shù)周圍的this是什么,它的"this"就是什么。(注意我在this上加了引號(hào),是因?yàn)檫@個(gè)this并不是真的箭頭函數(shù)的this

而且不會(huì)被bind, call, apply所改變

var foo = ()=>{ return this };
foo() // window
var obj = { foo: ()=>this }
obj.foo(); // 由于定義的時(shí)候,周圍的環(huán)境是window,所以返回window
var obj = {
  foo(){
     var bar= ()=>{ return this };
     return bar();
  }
}
obj.foo(); // 由于定義bar的時(shí)候,周圍環(huán)境是obj,所以返回obj
var foo = obj.foo;
foo(); // 同理,這里是window
var foo = ()=>{ return this };
foo = foo.bind(1);
foo(); // window
foo.call(1); // window
foo.apply(1); // window
function Foo(){
  // 箭頭函數(shù)
  var a = ()=>{
    console.log(this === object); // true
  }

  // 對(duì)比普通函數(shù)
  var b = function(){
    console.log(this === window); // true
  }
  
  this.foo = function(){
    a(); b();
  }
}

var object = new Foo();
object.foo();
function Foo(){}
// window
Foo.prototype.foo = ()=>{ return this }
// window
var object = new Foo();
object.foo(); // 由于定義foo的時(shí)候,周圍環(huán)境是window,所以這里是window
小結(jié)

之前在上面寫的都是記憶方法,方便記憶,但是并沒有涉及到原理,這里大概點(diǎn)幾點(diǎn):

bind以后,即使再bind, apply, call也不會(huì)改變this的值

其實(shí)可以從bind的簡(jiǎn)單實(shí)現(xiàn)看出來:

Function.prototype.bind = Function.prototype.bind || function(context, ...args){
  var func = this;
  return function(...currentArgs){
    return func.apply(context, args.concat(currentArgs));
  }
}

所以,從這里看出來,bind以后,返回了一個(gè)新的function,所以即使以后再對(duì)返回的方法進(jìn)行bind, apply, call,也只是對(duì)新的function操作,并不影響func.apply(context...)中的context值。但是可以影響后面的參數(shù)值。

關(guān)于bind方法的介紹,今天看到顏海鏡、顏大的文檔,感覺對(duì)bind的了解加深一步,附上鏈接

關(guān)于箭頭函數(shù)的this

根據(jù)MDN的文檔,箭頭函數(shù)是不會(huì)生成自己的this的(還有arguments)。所以我才給上面的this加上雙引號(hào)。而且由于它本身并沒有this,所以才不會(huì)被apply,bind, call所改變。

那么在箭頭函數(shù)中調(diào)用的this是怎么來定的呢?

下面來自vajoy的一篇譯文

雖然箭頭函數(shù)沒有一個(gè)自己的this,但當(dāng)你在內(nèi)部使用了this的時(shí)候,他會(huì)指向最近一層作用域的this

function foo(){
  return ()=>{
    return ()=>{
      return ()=>{
        console.log("id:", this.id);
      }
    }
  }
}
foo.call({id:42})()()(); // id:42

這里只有一次this的綁定,也就是foo()的時(shí)候.

這些連接內(nèi)嵌的函數(shù)們都沒有生命他們自己的this,所以this.id的引用會(huì)簡(jiǎn)單地順著作用域鏈查找,一直查找到foo()函數(shù),他是第一處能找到一個(gè)確切存在的this的地方。

由于只是我根據(jù)經(jīng)驗(yàn)和資料自己總結(jié)的,所以不知道是否有疏忽或者遺漏,如果有問題的地方,歡迎提出。謝謝

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/81542.html

相關(guān)文章

  • JavaScript專題之函數(shù)記憶

    摘要:專題系列第十七篇,講解函數(shù)記憶與菲波那切數(shù)列的實(shí)現(xiàn)定義函數(shù)記憶是指將上次的計(jì)算結(jié)果緩存起來,當(dāng)下次調(diào)用時(shí),如果遇到相同的參數(shù),就直接返回緩存中的數(shù)據(jù)。 JavaScript 專題系列第十七篇,講解函數(shù)記憶與菲波那切數(shù)列的實(shí)現(xiàn) 定義 函數(shù)記憶是指將上次的計(jì)算結(jié)果緩存起來,當(dāng)下次調(diào)用時(shí),如果遇到相同的參數(shù),就直接返回緩存中的數(shù)據(jù)。 舉個(gè)例子: function add(a, b) { ...

    RobinTang 評(píng)論0 收藏0
  • Javascript中函數(shù)作為對(duì)象魅力

    摘要:賦予了函數(shù)非常多的特性,其中最重要的特性之一就是將函數(shù)作為第一型的對(duì)象。那就意味著在中函數(shù)可以有屬性,可以有方法,可以享有所有對(duì)象所擁有的特性。參考資料忍者秘籍關(guān)于函數(shù)我之前還寫過一篇中高階函數(shù)的魅力有興趣的話可以看一看。 Javascript賦予了函數(shù)非常多的特性,其中最重要的特性之一就是將函數(shù)作為第一型的對(duì)象。那就意味著在javascript中函數(shù)可以有屬性,可以有方法, 可以享有...

    morgan 評(píng)論0 收藏0
  • JavaScript設(shè)計(jì)模式(八):組合模式

    摘要:不同于其它靜態(tài)編程語言,實(shí)現(xiàn)組合模式的難點(diǎn)是保持樹對(duì)象與葉對(duì)象之間接口保持統(tǒng)一,可借助定制接口規(guī)范,實(shí)現(xiàn)類型約束。誤區(qū)規(guī)避組合不是繼承,樹葉對(duì)象并不是父子對(duì)象組合模式的樹型結(jié)構(gòu)是一種聚合的關(guān)系,而不是。 showImg(https://segmentfault.com/img/bVbu79V?w=800&h=600); 組合模式:又叫 部分整體 模式,將對(duì)象組合成樹形結(jié)構(gòu),以表示 部分...

    leon 評(píng)論0 收藏0
  • JavaScript 編程精解 中文第三版 七、項(xiàng)目:機(jī)器人

    摘要:來源編程精解中文第三版翻譯項(xiàng)目原文譯者飛龍協(xié)議自豪地采用谷歌翻譯置疑計(jì)算機(jī)能不能思考就相當(dāng)于置疑潛艇能不能游泳。這張圖將成為我們的機(jī)器人在其中移動(dòng)的世界。機(jī)器人在收到包裹時(shí)拾取包裹,并在抵達(dá)目的地時(shí)將其送達(dá)。這個(gè)機(jī)器人已經(jīng)快了很多。 來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項(xiàng)目原文:Project: A Robot 譯者:飛龍 協(xié)議:CC BY-NC-S...

    jas0n 評(píng)論0 收藏0
  • 忍者級(jí)別JavaScript函數(shù)操作

    摘要:我們需要知道的是,對(duì)于而言,匿名函數(shù)是一個(gè)很重要且具有邏輯性的特性。通常,匿名函數(shù)的使用情況是創(chuàng)建一個(gè)供以后使用的函數(shù)。截圖自忍者秘籍通過完善之前對(duì)匿名函數(shù)的粗略定義,我們可以修復(fù)解決這個(gè)問題。 從名字即可看書,此篇博客總結(jié)與《JavaScript忍者秘籍》。對(duì)于JavaScript來說,函數(shù)為第一類型對(duì)象。所以這里,我們主要是介紹JavaScript中函數(shù)的運(yùn)用。 系列博客地址:h...

    suemi 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<