別真以為JavaScript中func.call/apply/bind是萬能的!

来源:http://www.cnblogs.com/venoral/archive/2016/05/06/5466976.html
-Advertisement-
Play Games

自從學會call/apply/bind這三個方法後我就各種場合各種使用各種得心應手至今還沒踩過什麼坑,怎麼用?說直白點就是我自己的對象沒有某個方法但別人有,我就可以通過call/apply/bind去調用執行別人家的方法,不太懂具體用法的同學可移至MDN學習一下Function.prototype. ...


自從學會call/apply/bind這三個方法後我就各種場合各種使用各種得心應手至今還沒踩過什麼坑,怎麼用?說直白點就是我自己的對象沒有某個方法但別人有,我就可以通過call/apply/bind去調用執行別人家的方法,不太懂具體用法的同學可移至MDN學習一下Function.prototype.call() Function.prototype.apply() Function.prototype.bind() ,本文不講解使用,但是這三個方法並不是萬能的,並不一定會執行你想要的那個函數,因為可能要看函數中上下文對象的某些屬性特性值允許不允許改變,今天這個坑我就深深踩了一下,並探索了一番。

事件起因
在學習HTML5與類相關的擴充其中classList屬性時候,我瞭解到所有元素有classList這個屬性,並且這個屬性是新集合DOMTokenList的實例,在DOMTokenList.prototype上有一個add(value)方法,用來將給定字元串值添加到DOMTokenList實例列表中,並且即時反應到文檔頁面。如果在列表中value值已存在,就不添加了。我手賤尋思著自己用JS代碼把這個add方法大概實現一下(肯定沒人家JS引擎實現的好,我就實現個思路),但就這一實現才發現的這個坑。
(1).先來看這個add方法JS引擎給它設置的屬性特性值如何,是可寫的:

(2).然後實現我們的add函數

Object.defineProperty(DOMTokenList.prototype, 'add', {
   configureable : true,
   enumerable : true,
   writable : true,
   value : function(value){ 
      if([].indexOf.call(this, value)>=0) return;
      //加到classList類數組中
      [].push.call(this, value);
  }
});

是不是感覺沒什麼錯,我就是想把'classA'加入到DOMTokenList實例列表中,然而當測試後控制台會報錯


分析探討
這個報錯內容為"類型錯誤:不能設置改變[object Object]的那個特性值只有getter的length屬性"。意思就是[object Object]就是這裡的DOMTokenList.prototype,你換成childNodes它也同樣會給你提示出錯。

這裡push方法因為會改變數組的長度length,而且不止push,pop,unshift,shift的執行也會改變數組長度,經測試它們會報出同樣的錯。這裡的DOMTokenList.prototype.length獲取它的特性值為:

length的訪問器屬性中set特性被引擎設置為undefiend了,怪不得不能將value加到document.body.classList類數組中,因為引擎沒有給你提供set的介面函數啊,只給你提供get特性函數返回length長度。而我們平常使用的這個不會報錯是因為length屬性的value值是可變的,註意這裡其實是每個數組實例都有自己的length屬性。

不過好在DOMTokenList.prototype的length屬性configurable特性是true,意味著我可以自己寫length的set函數。現在整個過程就是:當實現我自己的add函數,因為add函數中調用到push操作,當執行push操作時會更新DOMTokenList.prototype.length(自動調用set函數),我可以在set函數中執行相關處理,先拿console測試一下是不是這個流程

Object.defineProperty(DOMTokenList.prototype, 'length', {
  set : function(){console.log('執行了')}
})


確實是的,而且現在不報錯了。但是我現在做的只是皮毛,'classA'還沒有加到document.body.classList中去呢,看看列表裡還只是"document":

不知道push函數引擎是怎麼實現的,這說明JS引擎好像就沒執行push函數,此方法行不通...不清楚它為什麼不執行push操作??

那還是乖乖轉化為數組再處理吧。常見的是將類數組轉化為真正的數組再做相關處理,方法挺多比如Array.from(),Array.prototype.slice.call()等等,不過這會改變原來類數組的類型啊,我就想問怎麼樣處理能給類數組添加項而不改變類數組的類型。數組實例的類型由它的__proto__決定,那就好辦了,不過得先設置classList屬性的一些特性項,JS引擎給的set是undefiend

最後重寫下來為:

Object.defineProperty(DOMTokenList.prototype, 'add', {
   configureable : true,
   enumerable : true,
   writable : true,
   value : function(value){ 
      if([].indexOf.call(this, value)>=0) return;
      //加到classList類數組中
      var newarr = Array.from(this);
      newarr.push(value);
      newarr.__proto__ = DOMTokenList.prototype;
      document.body.classList = newarr;
  }
});

Object.defineProperty(Element.prototype, 'classList', {
  set: function(value){
    console.log(value);
    //這裡怎麼處理不同元素的classList值是JS引擎的事,我實在是不會了
  }
})

不過這樣寫有點固定具體元素了繼續重寫為:

Object.defineProperty(DOMTokenList.prototype, 'add', {
   configureable : true,
   enumerable : true,
   writable : true,
   value : function(value, ele){ 
      if([].indexOf.call(this, value)>=0) return;
      //加到classList類數組中
      var newarr = Array.from(this);
      newarr.push(value);
      newarr.__proto__ = DOMTokenList.prototype;
      ele.classList = newarr;
  }
});

Object.defineProperty(Element.prototype, 'classList', {
  set: function(value){
    console.log(value);
    //這裡怎麼處理不同元素的classList值是JS引擎的事,我實在是不會了
  }
});

//使用
document.body.classList.add('classA', document.body);//["classA"]

 

總結
受限於JS引擎中元素實例的某些屬性是共用於其原型屬性的屬性值的set函數,雖然最後也沒有全部實現add函數的功能,我是實在不知道JS引擎中怎麼實現的set,get函數從而保證不同實例共用原型上同一屬性而且還保證不同實例的該屬性值不一樣,這估計得看引擎源碼了,心累...有知道的同學可以說一下思路嗎??還有對於我上面的分析部分我有兩點疑問:知道的大神求科普!!
(1).JS引擎好像就沒執行push函數,此方法行不通...不清楚它為什麼不執行push操作
(2).怎麼樣處理能給類數組添加項而不改變類數組的類型


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

-Advertisement-
Play Games
更多相關文章
  • 前言 這篇博客的目的是對項目中偶爾碰到的應用設計模式的代碼積累一下。 我認為針對於設計模式的學習,通過看書、看例子只能瞭解概念,而重要的是如何把概念應用到實際的項目開發中。 而且我想提醒大家的是,學習設計模式時,千萬不要拘泥於書上的內容,形成定式思維,這樣對於其他源碼理解會出現一層自然屏障,理解起來 ...
  • 今天是五.四青年節,祝大家節日快樂。看著今天這標題就有食欲,夏天到了,醋溜土豆絲和清炒苦瓜適合夏天吃,好吃不上火。這兩道菜大部分人都應該吃過,特別是醋溜土豆絲,作為“魯菜”的代表作之一更是為大眾所熟知,醋溜土豆絲,好吃不上火。清炒苦瓜這道菜好啊,更是夏天必備之良菜,其功效在此就不做過多贅述了。言歸正 ...
  • 備忘錄:在不破壞封裝的前提下,捕獲一個對象的內部狀態,併在該對象之外保存這個狀態,這樣以後就可將該對象恢復到原先保存的狀態。 舉例如下: Originator:發起人,負責創建一個備忘錄Memento,用以記錄當前時刻它的內部狀態,並可使用備忘錄恢復內部狀態。它根據需要決定Memento存儲Orig ...
  • 一、Javascript中的數據類型可以分為基本數據類型和複合數據類型兩種: (1)基礎數據類型有5種數據類型:Number、Boolean、Undefined、Null和String。 Number類型:整數和浮點數。NaN:Not a Number。這個數值用於本來要返回一個數值,但是卻未能放回 ...
  • × 目錄 [1]display [2]visibility [3]hidden[4]opacity[5]overflow[6]clip[7]transform[8]覆蓋[9]偏移 前面的話 在網頁製作中,元素的顯示隱藏是非常常見的需求。本文將介紹元素顯示隱藏的9種思路 思路一: display 對於 ...
  • 效果圖: 第一張使用label標簽,第二張沒有使用.. 使用label標簽,middle對齊方式的單選框下降了1px 而沒有使用label標簽,sub對齊方式的 卻 居 中 了 =_= 不太理解,困.. 留一記錄, 忘能解惑 代碼: 1.說明名字用<label>標簽 2.說明文字沒有使用<label ...
  • 一、字元串操作包括哪些? search 查找 <script> var str='abcdefd'; alert(str.search('a')); //返回0,從0開始 alert(str.search('bc')); //返回1,這個1是bc出現的位置,search只找第一次出現的位置 aler ...
  • 效果:http://hovertree.com/texiao/js/24/ 效果圖: 代碼如下: 轉自:http://hovertree.com/h/bjaf/jsdiaocha.htm 特效:http://www.cnblogs.com/roucheng/p/texiao.html ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...