摘要:框架加冕時代年橫空出世的前端框架的模塊機(jī)制的模塊機(jī)制相比老王老李的解決方案上增強了模塊的約束性,和幫助開發(fā)者劃分模塊外,最重要的是解決了模塊的運行時管理問題模塊的初始化順序問題和依賴的模塊自動初始化問題。
前端框架工程化之路
人類的發(fā)展動力源于一個“懶”字,就如現(xiàn)在的大前端正是史前那群“懶”而聰明的“切圖仔”進(jìn)了軟件工程的施工現(xiàn)場,懷揣著更少代碼、更少溝通、更少錯誤、更少維護(hù)的夢想奔襲而來。從框架齊放鬧革命到三大框架三足鼎立,從構(gòu)建工具爭鳴到Webpack一統(tǒng)江湖,從Javascript 遵循ES5長達(dá)7年統(tǒng)治到向ES6的自我進(jìn)化。前端的發(fā)展與它們的成功都離不開一個“術(shù)”,工程化。
模塊的進(jìn)化 在沒有框架的史前我們面臨的問題:
1.全局變量污染:各個文件的變量都是掛載到window對象上,污染全局變量。
2.變量重名:不同文件中的變量如果重名,后面的會覆蓋前面的,造成程序運行錯誤。
3.文件依賴順序:多個文件之間存在依賴關(guān)系,需要保證一定加載順序問題嚴(yán)重。
于是老王想出使用自執(zhí)行函數(shù)的方法去解決問題
var foo = (function(cNum){ var aStr = "aa"; var aNum = cNum + 1; return { aStr: aStr, aNum: aNum }; })(cNum);
前端最初始的模塊誕生了,這個模塊有問題嗎?有!雖然模塊內(nèi)部的變量對全局不可見了,但暴露出來的foo是一個全局變量,這樣的模塊多了全局變量也會很多。
老李在老王的辦法基礎(chǔ)上添加命名空間去解決問題:
app.util.modA = xxx; app.common.modA = xxx; app.tools.modA.format = xxx;
除了寫法丑陋外,這樣的模塊約束力極低,很容易遭到不遵守的開發(fā)者破壞,需要開發(fā)者有一定的劃分不同模塊的能力,更大的問題是需要人為的解決模塊加載、初始化等管理問題。
框架加冕時代2009年橫空出世的前端框架Angularjs的模塊機(jī)制
angular.module("ConfigModule").service("TextConfig", function () { this.headerText = { }; }); angular.module("HeaderModule", ["ConfigModule"]).controller("HeaderCtr", ["$scope", "TextConfig", function ($scope, textConfig) { $scope.headerText = textConfig.headerText; }]);
Angularjs的模塊機(jī)制相比老王、老李的解決方案上增強了模塊的約束性,和幫助開發(fā)者劃分模塊外,最重要的是解決了模塊的運行時管理問題(模塊的初始化順序問題和依賴的模塊自動初始化問題。再被多個模塊依賴的情況,模塊僅且只加載一次的問題,統(tǒng)一的輸入輸出api問題)。看似完美的方案,但仍有問題。
構(gòu)建工具輔政Angularjs的模塊機(jī)制只解決了運行時管理問題,但沒有解決模塊加載管理問題。這讓使用者不得不去鏈?zhǔn)皆陧撁嬉媚K文件。所以在那個時候出現(xiàn)了一些構(gòu)建工具與相應(yīng)的插件來幫助我們 比如Gulp、Grunt、插件 Browserify。
實際上Angularjs的模塊機(jī)制也只是一定程度的解決了運行時管理問題,了解的同學(xué)應(yīng)該知道在Angularjs里做模塊異步懶加載是件非常困難的事情。在Angular 2及以上版本加入了動態(tài)加載模塊的支持。其它框架,例如Vue組件(這里暫且把Vue的組件當(dāng)作模塊看待,后面會進(jìn)行區(qū)分)也加了相應(yīng)的支持, 這得益于框架的組件或模塊的factor機(jī)制的支持和Webpack code splitting功能的支持。
const foo = resolve => { require.ensure(["./Foo.vue"], () => { resolve(require("./Foo.vue")) }) } const router = new VueRouter({ routes: [ { path: "/foo", component: Foo } ] })一直在接近,但從未實現(xiàn)的組件化
前面談模塊化發(fā)展中談到了Vue的組件,組件是一種模塊,但又超越模塊。模塊是邏輯單元的封裝,讓開發(fā)及維護(hù)成本更低。那么組件則是更高一層的抽象,是一個業(yè)務(wù)單元的封裝,能夠獨立運行的軟件單元。組件需要解決的問題:隔離、去污染問題(模塊解決的隔離只限于js變量的隔離、而組件還需要解決css的隔離)、狀態(tài)管理問題、與其它組件通訊問題,生命周期問題,一個好的組件設(shè)計還需要遵循軟件設(shè)計的一些原則。
不得不改源碼的Jquery組件我們先來看看Jquery的年代組件長什么樣? 以前的代碼一般是用自執(zhí)行函數(shù)作為一個類,這里為方便理解,我用TS展示一下。
export class component { selector:Element; options:any static defaultOption = { "color": "red", "fontSize": "16px", "textDecoration":"none"} constructor(selector,opt) { this.selector = selector; this.options = $.extend({}, this.defaultOption, opt) } highlight () { return this. selector.css({ "color": this.options.color, "fontSize": this.options.fontSize, "textDecoration": this.options.textDecoration }); } }
組件使用者拿到這個組件并初始化,根據(jù)組件上層的一些交互,調(diào)用組件方法,改動組件內(nèi)部。 我們可以看到組件上層依賴這個組件,且依賴的是highlight()的具體實現(xiàn)。根據(jù)OCP原則,對擴(kuò)展開放,對修改關(guān)閉。當(dāng)我們需求變化時,比如我們的highlight需要把背景色改一下,只能對組件內(nèi)部邏輯做修改。很顯然這樣的組件不是一個好設(shè)計。
數(shù)據(jù)驅(qū)動讓組件高可用性進(jìn)入有前端框架的時代,Angular使用數(shù)據(jù)驅(qū)動改變視圖的狀態(tài),這是很大的一個進(jìn)步,數(shù)據(jù)驅(qū)動解耦了組件外層對組件的依賴關(guān)系,將真正的依賴拋向外層的傳給組件的數(shù)據(jù)(有點類似依賴倒置的意思),組件內(nèi)部負(fù)責(zé)根據(jù)數(shù)據(jù)的變化改變UI狀態(tài)。(React Virtual DOM 驅(qū)動視圖實際也是一種數(shù)據(jù)驅(qū)動,只是一個是找到數(shù)據(jù)最小粒度的變化直接改動對應(yīng)的視圖,一個是數(shù)據(jù)生成Virtual DOM找到最小粒度的Virtual DOM變化,改動對應(yīng)的視圖。本質(zhì)上都是數(shù)據(jù)驅(qū)動視圖)。
數(shù)據(jù)驅(qū)動視圖解耦了組件與組件的依賴問題。但同時引入了一個問題,狀態(tài)混亂問題。寫過Angularjs的同學(xué)應(yīng)該知道狀態(tài)混亂之痛,當(dāng)我們在Angularjs的多個組件依賴同一份數(shù)據(jù)時,當(dāng)一個組件樹中某一個組件將該數(shù)據(jù)更改時,整顆組件樹中使用該數(shù)據(jù)的組件都會跟著共振。但實際情況是,樹當(dāng)中有一部分組件不需要跟著某一次的數(shù)據(jù)變化而變化。
React、Angular使用Immutablejs強化單向數(shù)據(jù)流。這的確減輕了復(fù)雜度,但這種方式對于子組件想通過狀態(tài)變更驅(qū)動父組件、兄弟組件變化的情況,只能通過注冊事件通知的形式。首先這種形式會違背隔離性,有很高的耦合,組件內(nèi)部必須知道外部想要知道我會有什么變化,預(yù)留訂閱的鉤子。其次對于一顆組件樹跨了N層,極端點從葉子節(jié)點到根節(jié)點這樣一個通知在每層訂閱子組件事件,會顯得非常不合理。
于是衍生的一些Flux Redux庫的狀態(tài)集中式托管,讓一個組件的數(shù)據(jù)驅(qū)動視圖的變化,可以來源于任何一個組件樹節(jié)點,又不會讓變化成復(fù)雜的網(wǎng)狀拓?fù)浣Y(jié)構(gòu),而是成星型拓?fù)浣Y(jié)構(gòu)。
這個發(fā)展過程實際也是為了解耦合,讓組件更加獨立,就好似以前一個人的每一個日常活動都推送出去讓全人類知道,由于日常太過苦逼影起社會負(fù)面情緒暴增,影響太大了(無單向數(shù)據(jù)流的情況)。于是乎改為在推送前,大家先訂閱建立關(guān)系,建立關(guān)系則推送,大家發(fā)現(xiàn)這套太麻煩了,需要個體知道別人關(guān)心我哪個日常活動,家人想知道我吃的什么我建立一個曬圖發(fā)布方式,領(lǐng)導(dǎo)想知道工作情況,我建立一個寫周報發(fā)布方式,于是乎我不停的改變自己適應(yīng)社會(這就是一種耦合,單向數(shù)據(jù)流方式)。當(dāng)個體喪失人性后,終于想到了一個更好的辦法,做自己該做的事,把這種拔內(nèi)褲的事情交給社會,我在辦公室就給我采集認(rèn)真工作的照片,系統(tǒng)讓領(lǐng)導(dǎo)看還是讓其它同事看作為個體的我不需關(guān)心。我在吃大餐的時候你采集照片,系統(tǒng)要給哪些聯(lián)系人,這由我此刻處的環(huán)境下社會關(guān)系決定。這個系統(tǒng)就是狀態(tài)管理器,這個社會就是組件所處的環(huán)境。
組件進(jìn)化之殤組件的進(jìn)化從未停下腳步,例如css隔離問題從依靠項目wiki中制訂css命名規(guī)范到css Modules自動化解決css隔離問題。從Angularjs的混亂的網(wǎng)狀狀態(tài)管理到React、Angular使用Immutablejs強化單向數(shù)據(jù)流、和衍生的一些Flux Redux庫的狀態(tài)集中式托管。從Vue的簡單生命周期到2.0加入keep-alive、activated、deactivated使生命周期的增強。組件的發(fā)展一片欣欣向榮,但為什么我仍認(rèn)為還沒有實現(xiàn)組件化呢?
框架的出現(xiàn)讓組件擁有了其應(yīng)該有的特性,讓開發(fā)者無需再重復(fù)造輪子解決這些問題。但也引入了新的問題,組件的獨立性。
前面提到組件應(yīng)當(dāng)是一個獨立運行的軟件單元。而實際的情況是,組件只是在某框架體系下獨立運行的軟件單元。而工程化也是一個去底層服務(wù)的趨勢,我們可以看看近年的Docker技術(shù)、云服務(wù)的Serverless概念,都強調(diào)其無需關(guān)注底層執(zhí)行環(huán)境,想象一下如果某天我們開發(fā)一個頁面無論采用任何技術(shù)架構(gòu)與框架,都只需引入一個個Custom Element,把DOM Attribute作為API(或者拿著各團(tuán)隊發(fā)布的在線運行的一個組件地址),去組裝頁面即可。這就是近幾年前端的一個研究課題微前端技術(shù)。目前的微前端技術(shù)也有不錯的發(fā)展,利用Custom Element的實現(xiàn)方案,解決了一些基本的問題,如前面提到的:隔離性、狀態(tài)管理、通訊問題、生命周期問題、不依賴前端架構(gòu)體系問題。但作為能獨立服務(wù)端部署提供使用的一個component還有很多問題待解決,但這是一個組件化發(fā)展的方向。
這是一個不錯的微前端實現(xiàn)方案 https://micro-frontends.org/
工程化的烏托邦——規(guī)范化規(guī)范是工程的施工藍(lán)圖,保證產(chǎn)出的產(chǎn)品穩(wěn)健和易維護(hù)。如果沒有規(guī)范我們改一行代碼可能出現(xiàn)這樣的情況。
那是不是有了規(guī)范文檔,就好了?在趕著上線的高壓、高疲勞下狀態(tài),可能出現(xiàn)這樣的同事。
我們前面已經(jīng)提到過一些古老“法典”,如Jquery時代模塊定義的規(guī)則、css規(guī)范的規(guī)則,還有前面沒提到的項目結(jié)構(gòu)劃分、代碼書寫規(guī)范。 大家有沒有發(fā)現(xiàn)它們都在歷史的舞臺中消失或者說們不必在為規(guī)范的實現(xiàn)耗費精力。為什么?當(dāng)一個“法典”的受管控者和執(zhí)法者都是自身的時候,那法典也就成了空談。所以我們需要一個公正的執(zhí)法者——機(jī)器(自動化)。
終會讓項目wiki消失的自動化代碼規(guī)范,我們擁有jslint幫我們校驗,有編輯器插件幫我們根據(jù)規(guī)范自動格式化。
項目結(jié)構(gòu),我們有對應(yīng)的CLI幫我們生成。
模塊的定義,我們有框架幫助劃分解決運行時問題,有Webpack幫助解決加載問題,等等。
自動化并非真正讓規(guī)范消失,而是對規(guī)范的更加強制化和易實施化,達(dá)到“無約”自制的效果。
那些wiki里規(guī)范讓開發(fā)者要怎么怎么寫代碼的文章我覺得大可不必,話說:“能動手的不BB,能自動化的不文檔”。
而對于項目wiki里那些讓開發(fā)者如何與其它人合作寫代碼的文檔我也覺得大可不必。比如:我們與后端如何對接,我們可以使用YAPI這類工具,讓前后端對接口定義及數(shù)據(jù)結(jié)構(gòu)一目了然和保持實時穩(wěn)定性。
對于前端與前端之間如何互相調(diào)用模塊或組件,我們可以利用Typescript,讓模塊組件接口更加清晰和強類型帶來的穩(wěn)定性。
對于強類型帶來的其它好處我舉個例子:
這個是我寫無ts的vue項目時一個很低級的bug
export default { props: {}, data() { return { isFullscreen: false; } }, methods: { toggle() { this.isFullScreen = true; } } }
看似是個大小寫問題,但完全是可以避免,如果這個組件是個強類型,IDE(支持ts的)會推斷this類型,該字段是否聲明過給予校驗提示。這個是書寫上的帶來的好處。強類型還給我們帶來很多好處與方便,比如可以很快的了解一個模塊提供的API。可以在多模塊引用同一個數(shù)據(jù)時,某個時期對該數(shù)據(jù)結(jié)構(gòu)進(jìn)行一定調(diào)整后,能立刻知道那些陳舊的代碼哪些需要隨著這次改動一起調(diào)整等等。
除此之外TS還能在自動化文檔上起到輔助作用。我們可以看一下Angular的文檔自動生成工具有多棒~
https://compodoc.github.io/co...
demo: https://compodoc.github.io/co...
當(dāng)我們站在巨人的肩膀上時,從未覺得向前走一步是如此輕松...愿,未來的前端走得更輕松。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/100788.html
摘要:框架加冕時代年橫空出世的前端框架的模塊機(jī)制的模塊機(jī)制相比老王老李的解決方案上增強了模塊的約束性,和幫助開發(fā)者劃分模塊外,最重要的是解決了模塊的運行時管理問題模塊的初始化順序問題和依賴的模塊自動初始化問題。 前端框架工程化之路 人類的發(fā)展動力源于一個懶字,就如現(xiàn)在的大前端正是史前那群懶而聰明的切圖仔進(jìn)了軟件工程的施工現(xiàn)場,懷揣著更少代碼、更少溝通、更少錯誤、更少維護(hù)的夢想奔襲而來。從框架...
某熊的技術(shù)之路指北 ? 當(dāng)我們站在技術(shù)之路的原點,未來可能充滿了迷茫,也存在著很多不同的可能;我們可能成為 Web/(大)前端/終端工程師、服務(wù)端架構(gòu)工程師、測試/運維/安全工程師等質(zhì)量保障、可用性保障相關(guān)的工程師、大數(shù)據(jù)/云計算/虛擬化工程師、算法工程師、產(chǎn)品經(jīng)理等等某個或者某幾個角色。某熊的技術(shù)之路系列文章/書籍/視頻/代碼即是筆者蹣跚行進(jìn)于這條路上的點滴印記,包含了筆者作為程序員的技術(shù)視野、...
摘要:一步,兩步,三步四步五步,就這樣到達(dá)了人生的巔峰傳統(tǒng)前端生態(tài)初級不使用打包中間處理工具,手工處理圖片等資源掌握以下知識點基礎(chǔ)結(jié)構(gòu),基礎(chǔ)樣式,基礎(chǔ)語法框架,系列插件框架,等基礎(chǔ)插件,等其他移動端適配,瀏覽器兼容,瀏覽器調(diào)試等恭喜完成新手村修 一步,兩步,三步四步五步,就這樣到達(dá)了人生的巔峰~ 傳統(tǒng)前端生態(tài)-初級 不使用打包、中間處理工具,手工處理js、css、圖片等資源 掌握以下知識點:...
閱讀 3118·2021-11-23 09:51
閱讀 1983·2021-09-09 09:32
閱讀 1094·2019-08-30 15:53
閱讀 2965·2019-08-30 11:19
閱讀 2475·2019-08-29 14:15
閱讀 1442·2019-08-29 13:52
閱讀 559·2019-08-29 12:46
閱讀 2826·2019-08-26 12:18