自有前端工程師這個稱謂以來,前端的發展可謂是突飛猛進。相比較已經很是成熟的其餘領域,前端雖是後起之秀,但其野蠻生長是其餘領域不能比的。雖然前端技術飛快發展,可是前端總體的工程生態並無同步跟進。目前絕大多數的前端團隊仍然使用很是原始的「切圖(FE)->套模板(RD)」的開發模式,這種模式下的前端開發雖然說不是刀耕火種的原始狀態,可是效率很是低下。javascript
前端的工程化問題與傳統的軟件工程雖然有所不一樣,可是面臨的問題是同樣的。咱們首先回顧一下傳統的軟件開發流程模型:
php
上圖中的運行和維護並非串行關係,也並不是絕對的並行關係。維護貫穿從編碼到運行的整個流程。css
若是說計算機科學要解決的是系統的某個具體問題,或者更通俗點說是面向編碼的,那麼工程化要解決的是如何提升整個系統生產效率。因此,與其說軟件工程是一門科學,不如說它更偏向於管理學和方法論。html
軟件工程是個很寬泛的話題,每一個人都有本身的理解。以上是筆者我的的理解,僅供參考。前端
具體到前端工程化,面臨的問題是如何提升編碼->測試->維護階段的生產效率。java
可能會有人認爲應該包括需求分析和設計階段,上圖展現的軟件開發模型中,這兩個階段具體到前端開發領域,更恰當的稱謂應該是功能需求分析和UI設計,分別由產品經理和UI工程師完成。至於API需求分析和API設計,應該包括在編碼階段。node
要解決前端工程化的問題,能夠從兩個角度入手:開發和部署。python
從開發角度,要解決的問題包括:git
這兩個問題的解決方案有兩點:程序員
從部署角度,要解決的問題主要是資源管理,包括:
要解決上述問題,須要引入構建/編譯階段。
開發規範的目的是統一團隊成員的編碼規範,便於團隊協做和代碼維護。開發規範沒有統一的標準,每一個團隊能夠創建本身的一套規範體系。
值得一提的是JavaScript的開發規範,尤爲是在ES2015愈來愈普及的局面下,保持良好的編碼風格是很是必要的。筆者推薦Airbnb的eslint規範。
不少人會混淆模塊化開發和組件化開發。可是嚴格來說,組件(component)和模塊(module)應該是兩個不一樣的概念。二者的區別主要在顆粒度方面。《Documenting Software Architectures》一書中對於component和module的解釋以下:
A module tends to refer first and foremost to a design-time entity. ... information hiding as the criterion for allocating responsibility to a module.
A component tends to refer to a runtime entity. ... The emphasis is clearly on the finished product and not on the design considerations that went into it.
In short, a module suggests encapsulation properties, with less emphasis on the delivery medium and what goest on at runtime. Not so with components. A delivered binary maintains its "separateness" throughout execution. A component suggests independently deployed units of software with no visibility into the development process.
簡單講,module側重的是對屬性的封裝,重心是在設計和開發階段,不關注runtime的邏輯。module是一個白盒;而component是一個能夠獨立部署的軟件單元,面向的是runtime,側重於產品的功能性。component是一個黑盒,內部的邏輯是不可見的。
用通俗的話講,模塊能夠理解爲零件,好比輪胎上的螺絲釘;而組件則是輪胎,是具有某項完整功能的一個總體。具體到前端領域,一個button是一個模塊,一個包括多個button的nav是一個組件。
模塊和組件的爭論由來已久,甚至某些編程語言對二者的實現都模糊不清。前端領域也是如此,使用過bower的同行知道bower安裝的第三方依賴目錄是bower_component
;而npm安裝的目錄是node_modules
。也不必爲了這個爭得頭破血流,一個團隊只要統一思想,保證開發效率就能夠了。至因而命名爲module仍是component都無所謂。
筆者我的傾向組件黑盒、模塊白盒這種思想。
隨着web應用規模愈來愈大,模塊/組件化開發的需求就顯得愈來愈迫切。模塊/組件化開發的核心思想是分治,主要針對的是開發和維護階段。
關於組件化開發的討論和實踐,業界有不少同行作了很是詳細的介紹,本文的重點並不是關注組件化開發的詳細方案,便再也不贅述了。筆者收集了一些資料可供參考:
嚴謹地講,構建(build)和編譯(compile)是徹底不同的兩個概念。二者的顆粒度不一樣,compile面對的是單文件的編譯,build是創建在compile的基礎上,對所有文件進行編譯。在不少Java IDE中還有另一個概念:make。make也是創建在compile的基礎上,可是隻會編譯有改動的文件,以提升生產效率。本文不探討build、compile、make的深層運行機制,下文所述的前段工程化中構建&編譯階段簡稱爲構建階段。
在討論具體如何組織構建任務以前,咱們首先探討一下在整個前端工程系統中,構建階段扮演的是什麼角色。
首先,咱們看看目前這個時間點(2016年),一個典型的web先後端協做模式是什麼樣的。請看下圖:
上圖是一個比較成熟的先後端協做體系。固然,目前因爲Node.js的流行開始普及大前端的概念,稍後會講述。
自Node.js問世以來,前端圈子一直傳播着一個詞:顛覆。前端工程師要藉助Node.js顛覆以往的web開發模式,簡單說就是用Node.js取代php、ruby、python等語言搭建web server,在這個顛覆運動中,JavaScript是前端工程師的信心源泉。咱們不討論Node.js與php們的對比,只在可行性這個角度來說,大前端這個方向吸引愈來愈多的前端工程師。
其實大前端也能夠理解爲全棧工程師,全棧的概念與編程語言沒有相關性,核心的競爭力是對整個web產品從前到後的理解和掌握。
那麼在大前端模式下,構建又是扮演什麼角色呢?請看下圖:
大前端體系下,前端開發人員掌握着Node.js搭建的web server層。與上文提到的常規前端開發體系下相比,省略了mock server的角色,可是構建在大前端體系下的做用並無發生改變。也就是說,不管是大前端仍是「小」前端,構建階段在兩種模式下的做用徹底一致,構建的做用就是對靜態資源以及模板進行處理,換句話說:構建的核心是資源管理。
前端的資源能夠分爲靜態資源和模板。模板對靜態資源是引用關係,二者相輔相成,構建過程當中須要對兩種資源使用不一樣的構建策略。
目前仍然有大多數公司將模板交由後端開發人員控制,前端人員寫好demo交給後端程序員「套模板」。這種協做模式效率是很是低的,模板層交由前端開發人員負責可以很大程度上提升工做效率。
靜態資源包括js、css、圖片等文件,目前隨着一些新規範和css預編譯器的普及,一般開發階段的靜態資源是:
構建階段在處理這些靜態文件時,基本的功能應包括:
以上提到的幾個功能能夠說是爲了彌補瀏覽器自身功能的缺陷,也能夠理解爲面向語言自己的,咱們能夠將這些功能統稱爲預編譯。
除了語言自己,靜態資源的構建處理還須要考慮web應用的性能因素。好比開發階段使用組件化開發模式,每一個組件有獨立的js/css/圖片等文件,若是不作處理每一個文件獨立上線的話,無疑會增長http請求的數量,從而影響web應用的性能表現。針對諸如此類的問題,構建階段須要包括如下功能:
以上幾個功能除了壓縮是徹底自動化的,其餘兩個功能都須要人工的配置。好比爲了提高首屏渲染性能,開發人員在開發階段須要儘可能減小同步依賴文件的數量。
以上提到的全部功能能夠理解爲工具層面的構建功能。
以上提到的構建功能只是構建工具的基本功能。若是停留在這個階段,那麼也算是個及格的構建工具了,但也僅僅停留在工具層面。對比目前較流行的一些構建產品,好比fis,它具有以上所得的編譯功能,同時提供了一些機制以提升開發階段的生產效率。包括:
咱們也能夠將上面提到的功能理解爲平臺層面的構建功能。
模板與靜態資源是容器-模塊關係。模板直接引用靜態資源,通過構建後,靜態資源的改動有如下幾點:
其實url包括文件名的改動,之因此將二者分開論述是爲了讓讀者區分CDN與構建對資源的不一樣影響。
對於模板的構建宗旨是在靜態資源url和文件名改變後,同步更新模板中資源的引用地址。
如今有種論調是脫離模板的依賴,html由客戶端模板引擎渲染,簡單說就是文檔內容由JavaScript生成,服務端模板只提供一個空殼子和基礎的靜態資源引用。這種模式愈來愈廣泛,一些較成熟的框架也驅動了這個模式的發展,好比React、Vue等。但目前大多數web產品爲了提升首屏的性能表現,仍然沒法脫離對服務端渲染的依賴。因此對模板的構建處理仍然頗有必要性。
具體的構建策略根據每一個團隊的狀況有所差別,好比有些團隊中模板由後端工程師負責,這種模式下fis的資源映射表機制是很是好的解決方案。本文不討論具體的構建策略,後續文章會詳細講述。
模板的構建是工具層面的功能。
構建能夠分爲工具層面和平臺層面的功能:
一個完整的前端工程體系應該包括:
開發規範和組件化開發面向的開發階段,宗旨是提升團隊協做能力,提升開發效率並下降維護成本。
構建工具和平臺解決了web產品一系列的工程問題,旨在提升web產品的性能表現,提升開發效率。
隨着Node.js的流行,對於前端的定義愈來愈寬泛,在整個web開發體系中。前端工程師的角色愈來愈重要。本文論述的前端工程體系沒有涉及Node.js這一層面,當一個團隊步入大前端時代,前端的定義已經不只僅是「前端」了,我想Web工程師這個稱號更合適一些。
以前跟一位前端架構師討論構建中對於模塊化的處理時,他提到一個頗有意思的觀點:所謂的壓縮打包等爲了性能作出的構建,實際上是受限於客戶端自己。試想,若是將來的瀏覽器支持大規模併發請求、網絡延遲小到微不足道,咱們還須要壓縮打包嗎?
誠然,任何架構也好,策略也好,都是對當前的一種解決方案,並非一條條鐵律。脫離了時代,任何技術討論都沒有意義。