中文字幕日韩精品一区二区免费_精品一区二区三区国产精品无卡在_国精品无码专区一区二区三区_国产αv三级中文在线

如何改變函數(shù)的this指向

本篇內(nèi)容介紹了“如何改變函數(shù)的this指向”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

成都創(chuàng)新互聯(lián)公司主營寧河網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營網(wǎng)站建設(shè)方案,重慶APP開發(fā),寧河h5小程序開發(fā)搭建,寧河網(wǎng)站營銷推廣歡迎寧河等地區(qū)企業(yè)咨詢

如果是函數(shù)聲明,那么函數(shù)內(nèi)部的this指向全局對象或者調(diào)用該函數(shù)的對象。

箭頭函數(shù)的this綁定的是父級作用域內(nèi)的this

使用call,apply,bind可以綁定this指向。其中bind是永久改變this指向并且不會立即執(zhí)行。apply,call臨時改變this指向并且立即執(zhí)行。

apply,call不同之處在于調(diào)用的參數(shù)形式:

call(thisArg, arg1?, arg2?, arg3?…)第一個參數(shù)代表this指向,剩下的參數(shù)就是原函數(shù)的參數(shù)。

apply(thisArg, argArray?)第一個參數(shù)代表this指向,第二個參數(shù)是以數(shù)組的形式傳遞參數(shù)。

/* bind, call, apply調(diào)用示例 */

var obj = {

x: 'obj',

getX: function() {

// 注意這里不能使用箭頭函數(shù)

return this.x;

}

};

// 調(diào)用 obj 對象上的函數(shù)屬性, this就會綁定為obj

// 'obj'

console.log(obj.getX());

var x = 'window';

function target(a, b, c) {

console.log(a, b, c);

return this.x;

}

/* bind */

// undefined undefined undefined

// 'window'

console.log(target());

const targetBound = target.bind(obj, 1); // 此時targetBound就是一個改變了this指向的target

// 1 2 3

// 'obj'

console.log(targetBound(2, 3));

/* call */

// 4 5 6

// 'obj'

console.log(target.call(obj, 4, 5, 6));

/* apply */

// 7 8 9

// 'obj'

console.log(target.apply(obj, [7, 8, 9]));

自定義實現(xiàn)call,apply,bind

這三個函數(shù)都有很強(qiáng)的錯誤處理功能,假如傳入的thisArg是一個基本類型,也會被使用包裝類替換,雖然不會報錯,但是不推薦這樣寫,盡量傳遞復(fù)雜類型的變量作為thisArg。

自定義實現(xiàn)myBind,myCall,myApply

// 先定義一個函數(shù)將任意類型都轉(zhuǎn)換成對象,用于指向this

function anyToObj(value) {

  // symbol, bigint 就不判斷了,直接在default中

  switch (typeof value) {

    case 'boolean':

      return new Boolean(value);

    case 'number':

      return new Number(value);

    case 'string':

      return new String(value);

    case 'undefined':

      return this;

    case 'object':

      if (value === null) return this; // 這里的this就指向了全局對象

      return value;

    default:

      return value;

  }

}

/* myBind */

Function.prototype.myBind = function (thisArg, …argArray) {

  // 防止出現(xiàn) const myBind = Function.prototype.myBind; myBind();

  if (typeof this !== 'function') {

    throw new TypeError('myBind must be called on a function');

  }

  const that = this; // this 就指向 f.myBind() 的 f

  const thatArg = anyToObj(thisArg); // 處理一下thisArg

  const myBound = function (…args) {

    // 指定唯一的鍵

    const tempKey = Symbol('__innerFunction__');

    thatArg.tempKey = that; // 將 that 函數(shù)賦值給一個對象的屬性

    let ret;

    if (this instanceof myBound) {

      // 調(diào)用 myBind 之后返回的是 myBound,假如調(diào)用 new myBound(),就會進(jìn)這個分支

      ret = new thatArg.tempKey(…argArray.concat(args));

    } else {

      // 這里是調(diào)用 myBound(),這樣調(diào)用 tempKey 方法里的 this 就指向了 thatArg

      ret = thatArg.tempKey(…argArray.concat(args));

    }

    // 不會對對象造成污染

    delete thatArg.tempKey;

    return ret;

  };

  return myBound;

};

/* myCall */

// 與 myBind 相比直接調(diào)用了

Function.prototype.myCall = function (thisArg, …argArray) {

  if (typeof this !== 'function') {

    throw new TypeError('myCall must be called on a function');

  }

  const thatArg = anyToObj(thisArg);

  const tempKey = Symbol('__innerFunction__');

  thatArg.tempKey = this;

  const ret = thatArg.tempKey(…argArray);

  delete thatArg.tempKey;

  return ret;

};

/* myApply */

// 與 myCall 相比,第二個參數(shù)希望是數(shù)組形式,多了個 CreateListFromArrayLike 用于轉(zhuǎn)化 argArray

Function.prototype.myApply = function (thisArg, argArray) {

  if (typeof this !== 'function') {

    throw new TypeError('myApply must be called on a function');

  }

  const CreateListFromArrayLike = function (val) {

    // 同樣缺乏 bigint 的處理,直接放在 default 中

    switch (typeof val) {

      case 'string':

      case 'number':

      case 'boolean':

      case 'symbol':

        throw new TypeError('CreateListFromArrayLike called on non-object');

      case 'object':

        if (Array.isArray(val)) return val;

        if (val === null) return [];

        return Array.from(val);

      default:

        return [];

    }

  };

  // 檢測 argArray 的類型

  const tempArgArray = CreateListFromArrayLike(argArray);

  const thatArg = anyToObj(thisArg);

  const tempKey = Symbol('__innerFunction__');

  thatArg.tempKey = this;

  const ret = thatArg.tempKey(…tempArgArray);

  delete thatArg.tempKey;

  return ret;

};

// 這樣 myBind,myCall,myApply就完成了,下面進(jìn)行測試

var obj = {

x: 'obj',

getX: function(a, b, c) {

console.log(a, b, c);

return this.x;

}

}

var x = 'window';

// 1 2 3

// 'obj'

console.log(obj.getX(1, 2, 3));

// target變成一個全局函數(shù),this 指向全局對象

const target = obj.getX;

// 1 2 3

// 'window'

console.log(target(1, 2, 3));

/* myBind */

const targetBound = target.myBind(obj, 1);

// 1 2 3

// 'obj'

console.log(targetBound(2, 3));

/* myCall */

// 4 5 6

// 'obj'

console.log(target.myCall(obj, 4, 5, 6));

/* myApply */

// 7 8 9

// 'obj'

console.log(target.myApply(obj, [7, 8, 9]));

“如何改變函數(shù)的this指向”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

新聞標(biāo)題:如何改變函數(shù)的this指向
網(wǎng)站URL:http://www.rwnh.cn/article18/ghcpdp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供App設(shè)計、網(wǎng)站內(nèi)鏈、企業(yè)網(wǎng)站制作、響應(yīng)式網(wǎng)站、定制開發(fā)、網(wǎng)頁設(shè)計公司

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

商城網(wǎng)站建設(shè)
隆子县| 三台县| 临潭县| 静宁县| 拉萨市| 石渠县| 鱼台县| 崇礼县| 建瓯市| 揭阳市| 新丰县| 通渭县| 广灵县| 河间市| 辉县市| 怀宁县| 中江县| 南通市| 高台县| 江山市| 和平区| 修水县| 长治县| 于都县| 罗甸县| 盖州市| 海南省| 澄迈县| 宜春市| 涡阳县| 二连浩特市| 岳西县| 全州县| 塔河县| 邵东县| 河东区| 习水县| 汉寿县| 赣榆县| 随州市| 泽普县|