精讀《現(xiàn)代 js 框架存在的根本原因

1 引言

深入思考為何前端需要框架,以及 web components 是否可以代替前端框架?

原文地址,建議先閱讀原文,或者閱讀概述。

2 概述

現(xiàn)在前端框架非常多了,如果讓我們回答 “為什么要用前端框架” 這個(gè)問題,你覺得是下面這些原因嗎?

  • 組件化。
  • 擁有強(qiáng)大的開源社區(qū)。
  • 擁有大量第三方庫解決大部分問題。
  • 擁有大量現(xiàn)成的第三方組件。
  • 擁有瀏覽器拓展/工具幫助快速 debug。
  • 友好的支持單頁應(yīng)用。

不,這些都不是根本原因,最多算前端框架的營銷手段。作者給出的最根本原因是:

解決 UI 與狀態(tài)同步的難題。

作者假設(shè)了一個(gè)沒有前端框架的項(xiàng)目,就像 Jquery 時(shí)代,我們需要手動(dòng)同步狀態(tài)與 UI。就像下面的代碼:

addAddress(address) {
  // state logic
  const id = String(Dat.now())
  this.state = this.state.concat({ address, id })

  // UI logic
  this.updateHelp()

  const li = document.createElement('li')
  const span = document.createElement('span')
  const del = document.createElement('a')
  span.innerText = address
  del.innerText = 'delete'
  del.setAttribute('data-delete-id', id)

  this.ul.appendChild(li)
  li.appendChild(del)
  li.appendChild(span)
  this.items[id] = li
}

首先更新效率是個(gè)問題,最大問題還是同步問題。試想多次與服務(wù)器交互,在同步過程中漏執(zhí)行了一步,會(huì)導(dǎo)致之后的 UI 與狀態(tài)逐漸脫節(jié)。

因?yàn)槲覀冎荒芤徊讲酵綘顟B(tài)與 UI,卻無法保證每個(gè)瞬間 UI 與狀態(tài)是完全同步的,任何一個(gè)疏忽都會(huì)導(dǎo)致 UI 與狀態(tài)脫節(jié),而我們除了不斷檢查 UI 與數(shù)據(jù)是否對應(yīng),毫無辦法。

所以現(xiàn)代框架最重要的幫助是保持 UI 與狀態(tài)的同步。

如何做到

有兩種思路:

  1. 組件級重渲染:比如 React,當(dāng)狀態(tài)改版后,映射出改變后的虛擬 DOM,最終改變當(dāng)前組件映射的真實(shí) DOM,這個(gè)過程被稱為 reconciliation。
  2. 監(jiān)聽修改:比如 Angluar 和 Vue.js,狀態(tài)改變直接觸發(fā)對應(yīng) DOM 節(jié)點(diǎn)中 value 值的變化。

這里稍微說明下,React 雖然是整體渲染,但在虛擬 DOM 作用下,效率不比 observable 低。observable 在值不能完整映射 UI 時(shí),也需要做更大范圍的 rerender。另外,Vue.js 與 Angluar 也早已采用了虛擬 DOM。

這三個(gè)框架已經(jīng)融會(huì)貫通,作者提到的兩種思路現(xiàn)在已經(jīng)是一種混合技術(shù)了。

那 web components 呢?

大家經(jīng)常會(huì)拿 React, Angluar, Vue.js 與 web components 做比較,可 web components 最大的問題就是,沒有解決 UI 與狀態(tài)同步。

web components 只提供了模版語法,自定義標(biāo)簽解決 html 的問題,并沒有給出一套狀態(tài)與 UI 同步的方法。

所以就算使用 web components,我們可能還需要一個(gè)框架做 UI 同步,比如 Vue.js 或者 stenciljs

作者還提供了一段簡短的 UI 狀態(tài)同步實(shí)例,這里略過。

最后給出了四點(diǎn)總結(jié):

  • 現(xiàn)代 js 框架主要在解決 UI 與狀態(tài)同步的問題。
  • 僅使用原生 js 難以寫出復(fù)雜、高效、又容易維護(hù)的 UI 代碼。
  • Web components 沒有解決這個(gè)主要問題。
  • 雖然使用虛擬 DOM 庫很容易造一個(gè)解決問題的框架,但不建議你真的這么做!

3 精讀

作者的核心觀點(diǎn)是,現(xiàn)代前端框架主要解決 UI 與狀態(tài)同步的問題,這是毫無疑問的,也提到了包括 web components 也依然沒有解決這個(gè)問題。

這可能是 web 開發(fā)最核心的問題了。

最初開發(fā)者的精力都在前端標(biāo)準(zhǔn)化上,誕生了一系列解決標(biāo)準(zhǔn)化問題的庫,最有知名度的是 jquery。當(dāng)前端進(jìn)入 react 時(shí)代后,可以看到精力從解決標(biāo)準(zhǔn)化到解決 web 規(guī)范與實(shí)踐的沖突,這個(gè)沖突正是作者說的問題。

前端三劍客

問題就出現(xiàn)在 html、js、css 三者分離上。

html、css、js 各是一套獨(dú)立的體系,但 js 又能同時(shí)控制 html 與 css,那為了解決同步問題,最好將控制權(quán)全部交給 js

這樣 web components 的問題也就好理解了,web components 解決的是 html 問題,注定與 js 無關(guān)。

html 官方規(guī)范估計(jì)很難出現(xiàn)現(xiàn)代框架的設(shè)計(jì)了,因?yàn)楣俜皆O(shè)計(jì)中前端三劍客是相互分離的方案,為了解決現(xiàn)階段前端框架的問題,html 必須由 js 完全接管,這幾乎就是 jsx,或者支持 template 語法的 html,可這與最初網(wǎng)頁設(shè)計(jì)思路是違背的。

html 是獨(dú)立的,甚至可以不依賴 js 運(yùn)行,這天然導(dǎo)致了 UI 與狀態(tài)同步這個(gè)難題。

為什么一定要用 js

html 不依賴 js 的設(shè)計(jì)可能已經(jīng)跟不上前端發(fā)展步伐了,也許 jsx 或者 template 才是真正的未來。

誠然,html 現(xiàn)在的設(shè)計(jì)可以在不支持 js 的瀏覽器執(zhí)行,但就在最近,所有現(xiàn)代瀏覽器都支持了 service worker,它是凌駕于 html 執(zhí)行時(shí)機(jī)之上的 js 腳本,甚至可以攔截 html 請求。一個(gè)不支持 js 的瀏覽器,可能也無法支持 service worker,禁用 js 的堅(jiān)持可能只剩下安全性保護(hù)。

而實(shí)際上現(xiàn)代 web 頁面都使用了 js 完全主導(dǎo)網(wǎng)頁渲染,所以這已經(jīng)從技術(shù)問題上升到了社會(huì)問題,如今禁用 js 的瀏覽器還有多少網(wǎng)頁可以正常訪問?除了某些超大型網(wǎng)站對禁用 js 狀態(tài)做了特殊優(yōu)化以外,現(xiàn)在幾乎沒有前端項(xiàng)目會(huì)考慮禁用 js 的情況了,因?yàn)槲覀儾粫?huì)假設(shè) React、Angluar、Vue.js 框架代碼無法運(yùn)行。

所以為什么不融合 html 與 js 呢?

既然事實(shí)上 UI 已經(jīng)與 js 綁定了,那 w3c 為何不將 jsx 或者 template 列為標(biāo)準(zhǔn)呢?也許為了向前兼容,規(guī)范永遠(yuǎn)也邁不出這一步吧。

幸運(yùn)的是,這并不妨礙現(xiàn)代前端框架的大量普及,而且勢不可擋。

4 總結(jié)

也許 UI 與狀態(tài)同步的問題是前端發(fā)展的最大阻力,雖然現(xiàn)代化框架已經(jīng)解決了這個(gè)問題,但 w3c 標(biāo)準(zhǔn)卻一直無法往這個(gè)方向發(fā)力,導(dǎo)致 web 的下一個(gè)發(fā)展方向難以依靠標(biāo)準(zhǔn)規(guī)范來推動(dòng)。前端日新月異的發(fā)展,很大一部分是規(guī)范的發(fā)展帶來的,而現(xiàn)在我們進(jìn)入了一個(gè)由工業(yè)化領(lǐng)導(dǎo)的時(shí)代,規(guī)范很可能永遠(yuǎn)也跟不上來,隨之而來的是工業(yè)化社區(qū)也難以做進(jìn)一步突破。

前端不僅是 web,或者也許下一個(gè)突破并不在 web,而是 ar/vr 或者下一個(gè)人機(jī)交互場景。同樣,web 也不僅是前端三劍客,如果認(rèn)為 React、Angluar、Vue.js 帶來的工業(yè)化規(guī)范就是新的規(guī)范,前端才有動(dòng)力向后發(fā)展,比如基于虛擬 DOM 的新框架、新語言。

所以筆者推導(dǎo)出現(xiàn)代前端開發(fā)的本質(zhì),是將 js、html 的平行關(guān)系變成了 js 包含 html 的關(guān)系,正如上面所說,這可能背離了 w3c 的初衷,但這就是現(xiàn)在的潮流。

最后總結(jié)一下觀點(diǎn):

  1. 也是原作者的,現(xiàn)代 js 框架主要在解決 UI 與狀態(tài)同步的問題。
  2. 傳統(tǒng)的前端三劍客正面臨著進(jìn)一步發(fā)展乏力的危機(jī)。
  3. 現(xiàn)代前端框架正在告訴我們新的三劍客:js(虛擬 dom、虛擬 css)。

5 更多討論

討論地址是:精讀《現(xiàn)代 js 框架存在的根本原因》 · Issue #84 · dt-fe/weekly

如果你想?yún)⑴c討論,請點(diǎn)擊這里,每周都有新的主題,周末或周一發(fā)布。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,247評論 6 543
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,520評論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,362評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,805評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,541評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,896評論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,887評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,062評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,608評論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,356評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,555評論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,077評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,769評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,175評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,489評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,289評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,516評論 2 379

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,737評論 25 708
  • 十一的時(shí)候,抖抖拋棄了我,我內(nèi)心很憤恨。就算他后來哄我,我心里也極為不快。我的不高興是發(fā)自肺腑,銘記于心的。...
    宛如初見v宛若新生閱讀 281評論 0 0
  • 時(shí)間真的很快,我一直以為70天會(huì)很久,自己可能堅(jiān)持不下來,可能會(huì)在某一天偷懶不打卡,結(jié)果我沒有偷懶……最后一天馬上...
    放羊的小娃娃閱讀 200評論 6 1
  • “好,我們今日一別,讓我們十年后再相間!” “我們一定會(huì)達(dá)成心中的理想的!” “不見不散!” 十年轉(zhuǎn)瞬即逝,如今已...
    金猊閱讀 727評論 0 2
  • 說我過者如良醫(yī)。譽(yù)我善者如鴆酒。千古道脈期我者。為明師良友。眼前活計(jì)誘我者。為惡友魔黨。看一經(jīng)思與身心當(dāng)下相...
    平等一照閱讀 1,849評論 0 0