日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關閉右側工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
拋開Vue、React、JQuery這類第三方js,我們該怎么寫代碼?

第三方js的現(xiàn)狀

成都創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供達川網(wǎng)站建設、達川做網(wǎng)站、達川網(wǎng)站設計、達川網(wǎng)站制作等企業(yè)網(wǎng)站建設、網(wǎng)頁設計與制作、達川企業(yè)網(wǎng)站模板建站服務,10年達川做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡服務。

無論是新入行的小白還是有經(jīng)驗的開發(fā)者,前端圈里的人一定聽過這類第三方js的大名。

一方面是因為它們實在太火了:

  • 各種文章對框架進行對比、源碼解析以。
  • GitHub 上 star 數(shù)量高速增長。
  • 各種針對框架的培訓課程層出不窮。
  • ……

另一方面是因為用它們開發(fā)非常方便:

  • 利用腳手架工具幾行命令就可以快速搭建項目。
  • 減少大量的重復代碼,結構更加清晰,可讀性強。
  • 有豐富的UI庫和插件庫。
  • ……

但是一則 GitHub 放棄使用 JQuery 的消息讓我開始思考:

第三方js除了帶來便利之外還有哪些副作用?

拋棄第三方js我們還能寫出高效的代碼嗎?

第三方js的副作用

雪球滾起來

如果現(xiàn)在讓你開發(fā)一個項目,你會怎么做?

假設你熟悉的是React,那么用可以用create-react-app快速搭建一個項目。

  • 很好,react、react-dom、react-router-dom 已經(jīng)寫入了package.json,不過事情還沒完。
  • http請求怎么處理呢?引入axios吧。
  • 日期怎么處理?引入 moment 或 day 吧。
  • ……

要知道,這種“拿來主義”是會“上癮”的,所以第三方依賴就像一個滾動的雪球,隨著開發(fā)不斷增加,***所占體積越來越大。

如果用 webpack-bundle-analyzer 工具來分析項目的話,會發(fā)現(xiàn)項目代碼大部分體積都在node_modules目錄中,也就意味著都是第三方js,典型的二八定律(80%的源代碼只占了編譯后體積的20%)。

類似下面這張圖:

于是不得不開始優(yōu)化,比如治標不治本的code split(代碼體積并沒有減小,只是拆分了),比如萬試萬難靈的tree shaking(你確定shaking之后的代碼都只有你真正依賴的代碼?),優(yōu)化效果有限不說,更糟糕的是依賴的捆綁。

比如ant-design的模塊的日期組件依賴了moment,那我們在使用它的時候moment就被引入了。

而且我即使發(fā)現(xiàn)體積更小的dayjs可以基本取代moment的功能,也不敢引入,因為替換它日期組件會出問題,同時引入又增加了項目體積。

有些第三方js被合稱之為“全家桶”,這種叫法讓我想起了現(xiàn)在PC端的一些工具軟件,本來你只想裝一個電腦管家,結果它不斷彈窗提示你電腦不安全,建議你安裝一個殺毒軟件,又提示你軟件很久沒更新,提示你安裝某某軟件管家…..

本來只想裝一個,結果裝了全家。

工具馴化

如果你注意觀察,在這些第三方js的使用者中,會看到這樣一些現(xiàn)象:

  • 排他。一些使用 MV* 框架的開發(fā)者很喜歡站隊進行討論,比如喜歡用 VueJS 的開發(fā)者很可能會吐槽 ReactJS,喜歡 Angular 的開發(fā)者會噴 VueJS。
  • 浮躁。一些經(jīng)驗并不豐富的開發(fā)者會覺得:使用JavaScript操作DOM多么低效,直接來個第三方js雙向數(shù)據(jù)綁定好了。自己寫XMLHTTPRequest發(fā)送請求多么麻煩,來第三方js直接調(diào)用好了。
  • 局限。一些面試者以為自己熟悉某種第三方js之后就覺得自己技術不錯(甚至很多時候這種“熟悉”還要打上引號),大有掌握了某種第三方js就掌握了前端之意。

這些第三方js本來是為了提升開發(fā)效率的工具,卻不知不覺地把開發(fā)者馴化了,讓其產(chǎn)生了依賴。

如果每次讓你開發(fā)新項目,你不得不依賴第三方js提供的腳手架來搭建項目,然后才能開始寫代碼。

那么很可能你已經(jīng)形成工具思維,就像手里拿著錘子,是什么都是釘子,你處理問答的方式,看問題的角度很可能會受此局限。

同時也意味著你正在離底層原生編碼越來越遠,越不熟悉原生API,你就越只能依賴第三方js,如此循環(huán)往復。

怎么打破這種狀況?

先推薦張鑫旭的一篇文章《不破不立的哲學與個人成長》,當然就是放棄它們。

這里需要注意的是,我所說的放棄并不是所有項目都自己寫框架,這樣在效率上而言是做不到的。

更推薦的而是在一些時間相對充裕、影響(規(guī)模)不大的項目中進行嘗試。

比如開發(fā)某個公司內(nèi)部使用的小工具,或者頁面數(shù)量不多的時間不緊張(看個人開發(fā)速度)的小項目。

用原生API進行開發(fā)的時候我們可以參考下面兩條建議。

理解精髓

雖然我們不使用任何第三方js,但是其原理及實現(xiàn)我們是可以學習,比如你知道實現(xiàn)數(shù)據(jù)綁定的方式有臟值檢測、以及Object.defineProperty,那么你在寫代碼的時候就可以使用它們,你會發(fā)現(xiàn)懂這些原理和真正使用起來還有不小的距離。

換個角度而言,這也可以進一步加深我們對第三方js的理解。

當然我們的目的并不是為了再造一個山寨版的js,而是適當?shù)亟Y合、刪減和優(yōu)化已有的技術和思想,為業(yè)務定制最合適的代碼。

文中提到的第三方js受歡迎很重要的一個原因是因為對DOM操作進行了優(yōu)化甚至是隱藏。

JQuery號稱是DOM操作的利器,將DOM封裝成JQ對象并擴展了API,而MV框架取代JQuery的原因是因為在DOM操作這條路上做得更絕,直接屏蔽了底層操作,將數(shù)據(jù)映射到模板上。

如果這些MV的思考方式還只是停留在DOM的層次上的話估計也無法發(fā)展到今天的規(guī)模。

因為屏蔽DOM只是簡化了代碼而已,要搭建大型項目還要考慮代碼組織的問題,就是抽象和復用。

這些第三方js選擇的方式就是“組件化”,把HTML、js和CSS封裝在一個具有獨立作用域的組件中,形成可復用的代碼單元。

下面我們通過不引入任何第三方js的情況下來進行實現(xiàn)。

無依賴實踐

web components

先來考慮組件化。

其實瀏覽器原生就支持組件化(web components),它由3個關鍵技術組成,我們先來快速了解一下。

Custom elements(自定義元素)

一組js API,允許自定義元素及其行為,然后可以在您的用戶界面中按照需要使用它們。

簡單示例:

 
 
 
  1. // 定義組件類
  2. class LoginForm extends HTMLElement {
  3.   constructor() {
  4.     super();
  5.     ...
  6.   }
  7. }
  8. // 注冊組件
  9. customElements.define('login-form', LoginForm);

Shadow DOM(影子DOM)

一組js API,創(chuàng)建一顆可見的DOM樹,這棵樹會附著到某個DOM元素上。

這棵樹的根節(jié)點稱之為shadow root,只有通過shadow root 才可以訪問內(nèi)部的shadow dom,并且外部的css樣式也不會影響到shadow dom上。

相當于創(chuàng)建了一個獨立的作用域。

常見的shadow root可以通過瀏覽器的調(diào)試工具進行查看:

簡單示例:

 
 
 
  1. // 'open' 表示該shadow dom可以通過js 的函數(shù)進行訪問
  2. const shadow = dom.attachShadow({mode: 'open'})
  3. // 操作shadow dom
  4. shadow.appendChild(h1);

HTML templates(HTML模板)

HTML模板技術包含兩個標簽:和 。

當需要在頁面上重復使用同一個 DOM結構時,可以用 template 標簽來包裹它們,然后進行復用。

slot標簽讓模板更加靈活,使得用戶可以自定義模板中的某些內(nèi)容。

簡單示例如下:

 
 
 
  1.   

    My paragraph

  2. // template的使用
  3. let template = document.getElementById('my-paragraph');
  4. let templateContent = template.content;
  5. document.body.appendChild(templateContent);
  6.   Let's have some different text!
  7.   Let's have some different text!

MDN上還提供了一些簡單的例子。這里來一個完整的例子:

 
 
 
  1. const str = `
  2.   
  3.   

    My default text

  4. `
  5. class MyParagraph extends HTMLElement {
  6.   constructor() {
  7.     super();
  8.     const template = document.createElement('template');
  9.     template.innerHTML = str;
  10.     const templateContent = template.content;
  11.     this.attachShadow({mode: 'open'}).appendChild(
  12.       templateContent.cloneNode(true)
  13.     );
  14.   }
  15. }
  16. customElements.define('my-paragraph', MyParagraph);

完整的組件

不過這樣的組件功能還太弱了,因為很多時候組件之間是需要有交互的,比如父組件向子組件傳遞參數(shù),子組件調(diào)用父組件回調(diào)函數(shù)。

因為它是HTML標簽,所以很自然地想到通過屬性來傳遞。而恰好組件也有生命周期函數(shù)來監(jiān)聽屬性的變化,看似***!

不過問題又來了,首先是性能問題,這樣會增加對dom的讀寫操作。其次是數(shù)據(jù)類型問題,HTML標簽上只能傳遞字符串這類簡單的數(shù)據(jù),而對于對象、數(shù)組、函數(shù)等這類復雜的數(shù)據(jù)就無能為力了。

你很可能想到對它們進行序列化和反序列化來實現(xiàn),一來是弄得頁面很不美觀(想象一個長度為100的數(shù)組參數(shù)被序列化后的樣子)。二來是操作復雜,不停地序列化和反序列化既容易出錯也增加性能消耗。三來是一些數(shù)據(jù)無法被序列化,比如正則表達式、日期對象等。

好在我們可以通過選擇器獲取DOM實例來傳遞參數(shù)。但是這樣的話就不可避免地操作DOM,這可不是個好的處理方式。

另一方面,就組件內(nèi)部而言,如果我們需要動態(tài)地將一些數(shù)據(jù)顯示到頁面上也需要操作DOM。

組件內(nèi)部視圖與數(shù)據(jù)地通信

將數(shù)據(jù)映射到視圖我們可以采用數(shù)據(jù)綁定的形式來實現(xiàn),而視圖的變化影響到數(shù)據(jù)可以采用事件的綁定的形式。

數(shù)據(jù)綁定

怎么楊將視圖和數(shù)據(jù)建立綁定關系,通常的做法是通過特定的模板語法來實現(xiàn),比如說使用指令。

例如用x-bind指令來將數(shù)據(jù)體蟲到視圖的文本內(nèi)容中。

臟值檢測的機制在性能上有損耗我們不考慮,那么剩下的就是利用Object.defineProperty這種監(jiān)聽屬性值變化的方式來實現(xiàn)。

同時需要注意的是,一個數(shù)據(jù)可以對應多個視圖,所以不能直接監(jiān)聽,而是要建立一個隊列來處理。

整理一下實現(xiàn)思路:

  1. 通過選擇器找出帶有x-bind屬性的元素,以及該屬性的值,比如
    的屬性值是text。
  2. 建立一個監(jiān)聽隊列dispatcher保存屬性值以及對應元素的處理函數(shù)。比如上面的元素監(jiān)聽的是text屬性,處理函數(shù)是this.textContent = value;
  3. 建立一個數(shù)據(jù)模型state,編寫對應屬性的set函數(shù),當值發(fā)生變化時執(zhí)行dispatcher中的函數(shù)。

示例代碼:

 
 
 
  1. // 指令選擇器以及對應處理函數(shù)
  2. const map = {
  3.   'x-bind'(value) {
  4.     this.textContent = undefined === value ? '' : value;
  5.   }
  6. };
  7. // 建立監(jiān)聽隊列,監(jiān)聽數(shù)據(jù)對象屬性值得變動,然后遍歷執(zhí)行函數(shù)
  8. for (const p in map) {
  9.   forEach(this.qsa(`[${p}]`), dom => {
  10.     const property = attr(dom, p).split('.').shift();
  11.     this.dispatcher[property] = this.dispatcher[property] || [];
  12.     const fn = map[p].bind(dom);
  13.     fn(this.state[property]);
  14.     this.dispatcher[property].push(fn);
  15.   });
  16. }
  17. for (const property in this.dispatcher) {
  18.   defineProperty(property);
  19. }
  20. // 監(jiān)聽數(shù)據(jù)對象屬性
  21. const defineProperty = p => {
  22.   const prefix = '_s_';
  23.   Object.defineProperty(this.state, p, {
  24.     get: () => {
  25.       return this[prefix + p];
  26.     },
  27.     set: value => {
  28.       if(this[prefix + p] !== value) {
  29.         this.dispatcher[p].forEach(fun => fun(value, this[prefix + p]));
  30.         this[prefix + p] = value;
  31.       }
  32.     }
  33.   });
  34. };

這里不是操作了DOM了嗎?

沒關系,我們可以把DOM操作放入基類中,那么對于業(yè)務組件就不再需要接觸DOM了。

小結:

這里使用VueJS同樣的數(shù)據(jù)綁定方式,但是由于數(shù)據(jù)對象屬性只能有一個 set 函數(shù),所以建立了一個監(jiān)聽隊列來進行處理不同元素的數(shù)據(jù)綁定,這種隊列遍歷的方式和AngularJS臟值檢測的機制有些類似,但是觸發(fā)機制不同、數(shù)組長度更小。

事件綁定

事件的綁定思路比數(shù)據(jù)綁定更簡單,直接在DOM元素上進行監(jiān)聽即可。

我們以click事件為例進行綁定,創(chuàng)建一個事件綁定的指令,比如x-click。

實現(xiàn)思路:

  1. 利用DOM選擇器找到帶有x-click屬性的元素。
  2. 讀取x-click屬性值,這時候我們需要對屬性值進行一下判斷,因為屬性值有可能是函數(shù)名比如x-click=fn,有可能是函數(shù)調(diào)用x-click=fn(a, true)。
  3. 對于基礎數(shù)據(jù)類型進行判斷,比如布爾值、字符串,并加入到調(diào)用參數(shù)列表中。
  4. 為DOM元素添加事件監(jiān)聽,當事件觸發(fā)時調(diào)用對應函數(shù),傳入?yún)?shù)。

示例代碼:

 
 
 
  1. const map = ['x-click'];
  2. map.forEach(event => {
  3.   forEach(this.qsa(`[${event}]`), dom => {
  4.     // 獲取屬性值
  5.     const property = attr(dom, event);
  6.     // 獲取函數(shù)名
  7.     const fnName = property.split('(')[0];
  8.     // 獲取函數(shù)參數(shù)
  9.     const params = property.indexOf('(') > 0 ? property.replace(/.*\((.*)\)/, '$1').split(',') : [];
  10.     let args = [];
  11.     // 解析函數(shù)參數(shù)
  12.     params.forEach(param => {
  13.       const p = param.trim();
  14.       const str = p.replace(/^'(.*)'$/, '$1').replace(/^"(.*)"$/, '$1');
  15.       if (str !== p) { // string
  16.         args.push(str);
  17.       } else if (p === 'true' || p === 'false') { // boolean
  18.         args.push(p === 'true');
  19.       } else if (!isNaN(p)) {
  20.         args.push(p * 1);
  21.       } else {
  22.         args.push(this.state[p]);
  23.       }
  24.     });
  25.     // 監(jiān)聽事件
  26.     on(event.replace('x-', ''), dom, e => {
  27.       // 調(diào)用函數(shù)并傳入?yún)?shù)
  28.       this[fnName](...params, e);
  29.     });
  30.   });
  31. });

對于表單控件的雙向數(shù)據(jù)綁定也很容易,即在建立數(shù)據(jù)綁定修改value,然后建立事件綁定監(jiān)聽input事件即可。

組件與組件之間的通信

解決完組件內(nèi)部的視圖與數(shù)據(jù)的映射問題我們來著手解決組件之間的通信問題。

組件需要提供一個屬性對象來接收參數(shù),我們設定為props。

父=>子,數(shù)據(jù)傳遞

父組件要將值傳入子組件的props屬性,需要獲取子組件的實例,然后修改props屬性。

這樣的話就不可避免的操作DOM,那么我們考慮將DOM操作法放在基類中進行。

那么問題來了,怎么找到哪些標簽是子組件,子組件有哪些屬性是需要綁定的?

可以通過命名規(guī)范和選擇其來獲取嗎?比如組件名稱都以cmp-開頭,選擇器支不支持暫且不說,這種要求既約束編碼命名,同時有沒有規(guī)范保證。

簡單地說就是沒有靜態(tài)檢測機制,如果有開發(fā)者寫的組件不是以cmp-開頭,運行時發(fā)現(xiàn)數(shù)據(jù)傳遞失敗檢查起來會比較麻煩。

所以可以在另一個地方對組件名稱進行采集,那就是注冊組件函數(shù)。

我們通過customElements.define函數(shù)來注冊組件,一種方式是直接對該函數(shù)進行重載,在注冊組件的時候記錄組件名稱,但是實現(xiàn)有些難度,而且對原生API函數(shù)修改難以保證不會對其它代碼產(chǎn)生影響。

所以折中的方式是對齊封裝,然后利用封裝的函數(shù)進行組件注冊。

這樣我們就可以記錄所有注冊的組件名了,然后創(chuàng)建實例來獲取對應props我們就解決了上面提出的問題。

同時在props對象的屬性上編寫set函數(shù)進行監(jiān)聽。

到了這一步還只完成了一半,因為我們還沒有把數(shù)據(jù)傳遞給子組件。

我們不要操作DOM的話那就只能利用已有的數(shù)據(jù)綁定機制了,將需要傳遞的屬性綁定到數(shù)據(jù)對象上。

梳理一下思路:

  1. 編寫子組件的時候建立props對象,并聲明需要被傳參的屬性, 比如this.props = {id: ''}。
  2. 編寫子組件的時候不通過原生customElements.define,而是使用封裝過的函數(shù),比如defineComponent來注冊,這樣可以記錄組件名和對應的props屬性。
  3. 父組件在使用子組件的時候進行遍歷,找出子組件和對應的props對象。
  4. 將子組件props對象的屬性綁定到父組件的數(shù)據(jù)對象state屬性上,這樣當父組件state屬性值發(fā)生變化時,會自動修改子組件props屬性值。

示例代碼:

 
 
 
  1. const components = {};
  2. /**
  3.  * 注冊組件函數(shù)
  4.  * @param {string} 組件(標簽)名
  5.  * @param {class} 組件實現(xiàn)類
  6.  */
  7. export const defineComponent = (name, componentClass) => {
  8.   // 注冊組件
  9.   customElements.define(name, componentClass);
  10.   // 創(chuàng)建組件實例
  11.   const cmp = document.createElement(name);
  12.   // 存儲組件名以及對應的props屬性
  13.   components[name] = Object.getOwnPropertyNames(cmp.props) || [];
  14. };
  15. // 注冊子組件
  16. class ChildComponent extends Component {
  17.   constructor() {
  18.     // 通過基類來創(chuàng)建模板
  19.     // 通過基類來監(jiān)聽props
  20.     super(template, {
  21.       id: value => {
  22.         // ...
  23.       }
  24.     });
  25.   }
  26. }
  27. defineComponent('child-component', ChildComponent);
  28. // 注冊父組件
  29. class ParentComponent extends Component {
  30.   constructor() {
  31.     super(template);
  32.     this.state.myId = 'xxx';
  33.   }
  34. }

上面的代碼中有很多地方可以繼續(xù)優(yōu)化,具體查看文末示例代碼。

子=>父,回調(diào)函數(shù)

子組件的參數(shù)要傳回給父組件,可以采用回調(diào)函數(shù)的形式。

比較麻煩的時候調(diào)用函數(shù)時需要用到父組件的作用域。

可以將父組件的函數(shù)進行作用域綁定然后傳入子組件props對象屬性,這樣子組件就可以正常調(diào)用和傳參了。

因為回調(diào)函數(shù)操作方式和參數(shù)不一樣,參數(shù)是被動接收,回調(diào)函數(shù)是主動調(diào)用,所以需要在聲明時進行標注,比如參考AngularJS指令的scope對象屬性的聲明方式,用“&”符號來表示回調(diào)函數(shù)。

理清一下思路:

  1. 子組件類中聲明props的屬性為回調(diào)函數(shù),如 this.props = {onClick:'&'}。
  2. 父組件初始化時,在模板上傳遞對應屬性, 如。
  3. 根據(jù)子組件屬性值找到對應的父組件函數(shù),然后將父組件函數(shù)綁定作用域并傳入。如childComponent.props.onClick = this.click.bind(this)。
  4. 子組件中調(diào)用父組件函數(shù), 如this.props.onClick(...)。

示例代碼:

 
 
 
  1. // 注冊子組件
  2. class ChildComponent extends Component {
  3.   constructor() {
  4.     // 通過基類來聲明回調(diào)函數(shù)屬性
  5.     super(template, {
  6.       onClick: '&'
  7.     });
  8.     ...
  9.     this.props.onClick(...);
  10.   }
  11. }
  12. defineComponent('child-component', ChildComponent);
  13. // 注冊父組件
  14. class ParentComponent extends Component {
  15.   constructor() {
  16.     super(template);
  17.   }
  18.   // 事件傳遞放在基類中操作
  19.   click(data) {
  20.     ...
  21.   }
  22. }

穿越組件層級的通信

有些組件需要子孫組件進行通信,層層傳遞會編寫很多額外的代碼,所以我們可以通過總線模式來進行操作。

即建立一個全局模塊,數(shù)據(jù)發(fā)送者發(fā)送消息和數(shù)據(jù),數(shù)據(jù)接收者進行監(jiān)聽。

示例代碼

 
 
 
  1. // bus.js
  2. // 監(jiān)聽隊列
  3. const dispatcher = {};
  4. /** 
  5.  * 接收消息
  6.  * name 
  7.  */
  8. export const on = (name, cb) => {
  9.   dispatcher[name] = dispatcher[name] || [];
  10.   const key = Math.random().toString(26).substring(2, 10);
  11.   // 將監(jiān)聽函數(shù)放入隊列并生成唯一key
  12.   dispatcher[name].push({
  13.     key,
  14.     fn: cb
  15.   });
  16.   return key;
  17. };
  18. // 發(fā)送消息
  19. export const emit = function(name, data) {
  20.   const dispatchers = dispatcher[name] || [];
  21.   // 輪詢監(jiān)聽隊列并調(diào)用函數(shù)
  22.   dispatchers.forEach(dp => {
  23.     dp.fn(data, this);
  24.   });
  25. };
  26. // 取消監(jiān)聽
  27. export const un = (name, key) => {
  28.   const list = dispatcher[name] || [];
  29.   const index = list.findIndex(item => item.key === key);
  30.   // 從監(jiān)聽隊列中刪除監(jiān)聽函數(shù)
  31.   if(index > -1) {
  32.     list.splice(index, 1);
  33.     return true;
  34.   } else {
  35.     return false;
  36.   }
  37. };
  38. // ancestor.js
  39. import {on} from './bus.js';
  40. class AncestorComponent extends Component {
  41.   constructor() {
  42.     super();
  43.     on('finish', data => {
  44.       //...
  45.     })    
  46.   }
  47. }
  48. // child.js
  49. class ChildComponent extends Component {
  50.   constructor() {
  51.     super();
  52.     emit('finish', data);
  53.   }
  54. }

總結

關于基類的詳細代碼可以參考文末的倉庫地址,目前項目遵循的是按需添加原則,只實現(xiàn)了一些基礎的操作,并沒有把所有可能用到的指令寫完。

所以還不足以稱之為“框架”,只是給大家提供實現(xiàn)思路以及編寫原生代碼的信心。


標題名稱:拋開Vue、React、JQuery這類第三方js,我們該怎么寫代碼?
當前鏈接:http://www.5511xx.com/article/dpoisds.html