摘要:回到純靜態(tài)頁(yè)面開(kāi)發(fā)階段,讓頁(yè)面不需要后端渲染也能跑起來(lái)。改造開(kāi)始本文著重介紹如何將靜態(tài)頁(yè)面改造成后端渲染需要的模板。總結(jié)在后端渲染的項(xiàng)目里使用多頁(yè)應(yīng)用架構(gòu)是絕對(duì)可行的,可不要給老頑固們嚇唬得又回到傳統(tǒng)前端架構(gòu)了。
本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。前言
原文地址:https://segmentfault.com/a/1190000008203380
如果您對(duì)本系列文章感興趣,歡迎關(guān)注訂閱這里:https://segmentfault.com/blog/array_huang
近年來(lái)前端領(lǐng)域發(fā)展迅猛,前后端分離早已成為業(yè)界共識(shí),各類(lèi)管控系統(tǒng)(to B)上個(gè)SPA什么的也不值一提,但唯獨(dú)偏展示類(lèi)的項(xiàng)目,為了SEO,始終還是需要依賴(lài)于服務(wù)器端渲染html。
我過(guò)往也曾嘗試為SPA彌補(bǔ)SEO,但現(xiàn)在看來(lái),效果雖然達(dá)到了,但工作量也大大地增加(因?yàn)楹蠖擞玫腜HP,不能做到前后同構(gòu))。
雖然無(wú)法改變依賴(lài)服務(wù)器端渲染這一現(xiàn)實(shí),但我們可以去勇敢地?fù)肀们岸说膱?jiān)船利炮(aka webpack),把服務(wù)器端模板層給啃下來(lái)!
前導(dǎo)知識(shí)webpack的基本使用
利用webpack生成HTML文件,及,構(gòu)建模板布局系統(tǒng)
你所在項(xiàng)目后端模板引擎(如我目前使用的Slim框架,模板引擎僅為原生PHP)
兩個(gè)階段整個(gè)前端項(xiàng)目,以本文主題的視角來(lái)看,可以分為兩個(gè)階段:
純靜態(tài)頁(yè)面開(kāi)發(fā)階段在這個(gè)階段里,一切開(kāi)發(fā)都跟靜態(tài)網(wǎng)站無(wú)二致,按UI稿切好頁(yè)面搞好交互,要用到ajax請(qǐng)求API的也盡管寫(xiě),跟后端的協(xié)作點(diǎn)僅在于API文檔。
傳統(tǒng)前端的工作也就到這里為止了,但對(duì)我們來(lái)說(shuō),目前的成果并不是我們最終的交付;因此,注意了,在這個(gè)階段我們是可以“偷懶”的,比如說(shuō),一些明顯應(yīng)該由服務(wù)器端循環(huán)生成的部分(商品列表、文章列表等),我們寫(xiě)一遍就OK了。
動(dòng)態(tài)頁(yè)面改造階段這就是所謂的“套頁(yè)面”,傳統(tǒng)來(lái)說(shuō)是由后端來(lái)做的,實(shí)際上后端也是苦不堪言,畢竟模板不是自己寫(xiě)的,有時(shí)還是需要改造一番,而這正是我們前端要大力爭(zhēng)取的活。
在這個(gè)階段里,我們的主要工作是按照后端模板引擎的規(guī)則來(lái)撰寫(xiě)模板變量占位符,當(dāng)然這里面也不會(huì)少了循環(huán)輸出和邏輯判斷,另外也可能需要用到后端定義的一些函數(shù),視項(xiàng)目需求而定。
在兩個(gè)階段里來(lái)回往返這兩個(gè)階段不一定是完全獨(dú)立的,有需要的話也是可以做到來(lái)回往返的。
那什么時(shí)候才叫做“有需要”呢?舉個(gè)例子,當(dāng)你把原先的靜態(tài)頁(yè)面都改造成需要后端渲染的頁(yè)面模板后,卻發(fā)現(xiàn)后端此時(shí)并未準(zhǔn)備好相應(yīng)的模板變量,而你此時(shí)又需要對(duì)頁(yè)面的UI部分進(jìn)行修改,那么你就很被動(dòng)了,因?yàn)楦暮玫倪@些頁(yè)面模板根本跑不起來(lái)了。有兩種解決方案:
參考API mock的思路,來(lái)個(gè)模板變量 mock,這就相當(dāng)于一直留在動(dòng)態(tài)頁(yè)面改造階段了。
回到純靜態(tài)頁(yè)面開(kāi)發(fā)階段,讓頁(yè)面不需要后端渲染也能跑起來(lái)。具體怎么做呢?
區(qū)分開(kāi)兩個(gè)階段,使用不同的webpack配置。
在我們構(gòu)建生成頁(yè)面的前端模板(注意分清與后端模板的區(qū)別),判斷(判斷依據(jù)看這里)本次執(zhí)行webpack打包是在哪個(gè)“階段”,繼而選擇是生成靜態(tài)(且完整)的element,還是帶有模板變量占位符的element。這樣一來(lái),我們就可以隨時(shí)選擇在不同的階段(或稱(chēng)環(huán)境)里進(jìn)行開(kāi)發(fā)了。
改造開(kāi)始本文著重介紹如何將靜態(tài)頁(yè)面改造成后端渲染需要的模板。
配合后端模板命名規(guī)則生成相應(yīng)模板文件不同項(xiàng)目因應(yīng)本身所使用的后端框架或是其它需求,對(duì)模板放置的目錄結(jié)構(gòu)也會(huì)有所不一樣,那么,如何構(gòu)建后端所需要的目錄結(jié)構(gòu)呢?
在靜態(tài)網(wǎng)頁(yè)階段,我習(xí)慣把html/css/js都按照所屬頁(yè)面歸到各自的目錄中(公用的css/js也當(dāng)然是放到公用目錄中),看HtmlWebpackPlugin配置:
pageArr.forEach((page) => { const htmlPlugin = new HtmlWebpackPlugin({ filename: `${page}/index.html`, // page變量形如"product/index"、"product/detail" template: path.resolve(dirVars.pagesDir, `./${page}/html.js`), chunks: [page, "commons/commons"], hash: true, xhtml: true, }); pluginsConfig.push(htmlPlugin); });
而在改造階段,則放到后端指定位置:
pageArr.forEach((page) => { const htmlPlugin = new HtmlWebpackPlugin({ filename: `../../view/frontend/${page}.php`, // 通過(guò)控制相對(duì)路徑來(lái)確定模板的根目錄 template: path.resolve(dirVars.pagesDir, `./${page}/html.js`), chunks: [page, "commons/commons"], hash: true, xhtml: true, }); pluginsConfig.push(htmlPlugin); });
此時(shí)我模板目錄結(jié)構(gòu)是這樣的:
│ ├─alert │ index.php │ ├─article │ detail.php │ index.php │ ├─index │ index.php │ ├─product │ detail.php │ index.php │ └─user edit-password.php modify-info.php
這里需要注意的是,我的前端項(xiàng)目目錄實(shí)際上是作為后端目錄里的一個(gè)子目錄來(lái)存放的,這樣才能依靠相對(duì)路徑來(lái)確定模板文件存放的根目錄位置。
處理站內(nèi)鏈接對(duì)于站內(nèi)鏈接,我建議在前端模板里使用一個(gè)函數(shù)來(lái)適配兩個(gè)階段:
{ /* 拼接系統(tǒng)內(nèi)部的URL */ constructInsideUrl(url, request, urlTail) { urlTail = urlTail || ""; let finalUrl = config.PAGE_ROOT_PATH + url; if (!config.IS_PRODUCTION_MODE) { finalUrl += "/index.html" + urlTail; return finalUrl; } return ` "${url}"), $isStaticize)?>`; }, };
在前端模板里這么用:
"> ">
這樣做,就能分別在靜態(tài)頁(yè)面階段和后端渲染階段生成相應(yīng)的超鏈接。再者,在后端渲染階段,我們生成出來(lái)的也不一定是一個(gè)完整的url,可以像我上述代碼一樣,生成調(diào)用后端函數(shù)的模板代碼,從而靈活滿(mǎn)足后端的一些需求(比如說(shuō),我的項(xiàng)目有靜態(tài)化的需求,那么,靜態(tài)化后的站內(nèi)鏈接跟動(dòng)態(tài)渲染的又會(huì)有所不同了)。
處理模板變量這一塊其實(shí)我要說(shuō)的不多,無(wú)非就是按照后端模板引擎的規(guī)則,輸出變量、循環(huán)輸出變量、判斷條件輸出變量、調(diào)用后端(模板引擎)函數(shù)調(diào)整輸出變量。
關(guān)鍵是,我們需要拿到一份模板變量文檔,跟API文檔類(lèi)似,它實(shí)際上也是一份前后端的數(shù)據(jù)協(xié)議。有了這份文檔,我們才能在后端未完工的情況下,進(jìn)入動(dòng)態(tài)頁(yè)面改造階段,并根據(jù)其中內(nèi)容實(shí)現(xiàn)模板變量 mock。
爭(zhēng)討模板布局渲染權(quán)關(guān)于利用模板布局系統(tǒng)對(duì)多個(gè)頁(yè)面共有的部分實(shí)現(xiàn)復(fù)用,在之前的文章里已經(jīng)提及了,我設(shè)計(jì)該系統(tǒng)的思路恰恰是來(lái)自于后端模板渲染。那么,在前后端均可以實(shí)現(xiàn)模板布局系統(tǒng)的前提下,我們應(yīng)如何抉擇呢?我的答案是,前端一定要吃下來(lái)!
從前端的角度來(lái)看:
我們?cè)?strong>純靜態(tài)頁(yè)面開(kāi)發(fā)階段的產(chǎn)物就已經(jīng)是一個(gè)個(gè)完整的頁(yè)面了,再要拆開(kāi)并不現(xiàn)實(shí)。
由于在webpack的輔助下這套模板布局系統(tǒng)功能相當(dāng)強(qiáng)大,因此并沒(méi)有給整個(gè)項(xiàng)目添加額外的成本。
從后端的角度來(lái)看:
服務(wù)器拼接多個(gè)HTML代碼段本身也是有成本(比如磁盤(pán)IO成本)的,倒不如渲染一個(gè)完整的頁(yè)面。
在公共組件的分治管理上不會(huì)有很大變化,只不過(guò)以前是一個(gè)一個(gè)組件渲染好后再拼在一起,而現(xiàn)在是把各個(gè)組件的數(shù)據(jù)整合在一起來(lái)統(tǒng)一渲染罷了。
總結(jié)在后端渲染的項(xiàng)目里使用webpack多頁(yè)應(yīng)用架構(gòu)是絕對(duì)可行的,可不要給老頑固們嚇唬得又回到傳統(tǒng)前端架構(gòu)了。
示例代碼諸位看本系列文章,搭配我在Github上的腳手架項(xiàng)目食用更佳哦(笑):Array-Huang/webpack-seed(https://github.com/Array-Huang/webpack-seed)。
附系列文章目錄(同步更新)webpack多頁(yè)應(yīng)用架構(gòu)系列(一):一步一步解決架構(gòu)痛點(diǎn)
webpack多頁(yè)應(yīng)用架構(gòu)系列(二):webpack配置常用部分有哪些?
webpack多頁(yè)應(yīng)用架構(gòu)系列(三):怎么打包公共代碼才能避免重復(fù)?
webpack多頁(yè)應(yīng)用架構(gòu)系列(四):老式j(luò)Query插件還不能丟,怎么兼容?
webpack多頁(yè)應(yīng)用架構(gòu)系列(五):聽(tīng)說(shuō)webpack連less/css也能打包?
webpack多頁(yè)應(yīng)用架構(gòu)系列(六):聽(tīng)說(shuō)webpack連圖片和字體也能打包?
webpack多頁(yè)應(yīng)用架構(gòu)系列(七):開(kāi)發(fā)環(huán)境、生產(chǎn)環(huán)境傻傻分不清楚?
webpack多頁(yè)應(yīng)用架構(gòu)系列(八):教練我要寫(xiě)ES6!webpack怎么整合Babel?
webpack多頁(yè)應(yīng)用架構(gòu)系列(九):總有刁民想害朕!ESLint為你阻擊垃圾代碼
webpack多頁(yè)應(yīng)用架構(gòu)系列(十):如何打造一個(gè)自定義的bootstrap
webpack多頁(yè)應(yīng)用架構(gòu)系列(十一):預(yù)打包Dll,實(shí)現(xiàn)webpack音速編譯
webpack多頁(yè)應(yīng)用架構(gòu)系列(十二):利用webpack生成HTML普通網(wǎng)頁(yè)&頁(yè)面模板
webpack多頁(yè)應(yīng)用架構(gòu)系列(十三):構(gòu)建一個(gè)簡(jiǎn)單的模板布局系統(tǒng)
webpack多頁(yè)應(yīng)用架構(gòu)系列(十四):No復(fù)制粘貼!多項(xiàng)目共用基礎(chǔ)設(shè)施
webpack多頁(yè)應(yīng)用架構(gòu)系列(十五):論前端如何在后端渲染開(kāi)發(fā)模式下夾縫生存
webpack多頁(yè)應(yīng)用架構(gòu)系列(十六):善用瀏覽器緩存,該去則去,該留則留
本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。
原文地址:https://segmentfault.com/a/1190000008203380
如果您對(duì)本系列文章感興趣,歡迎關(guān)注訂閱這里:https://segmentfault.com/blog/array_huang
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/81331.html
摘要:回到純靜態(tài)頁(yè)面開(kāi)發(fā)階段,讓頁(yè)面不需要后端渲染也能跑起來(lái)。改造開(kāi)始本文著重介紹如何將靜態(tài)頁(yè)面改造成后端渲染需要的模板。總結(jié)在后端渲染的項(xiàng)目里使用多頁(yè)應(yīng)用架構(gòu)是絕對(duì)可行的,可不要給老頑固們嚇唬得又回到傳統(tǒng)前端架構(gòu)了。 本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。原文地址:https://segmentfault.com/a/119000000820338...
摘要:本文首發(fā)于的技術(shù)博客實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。原文地址如果您對(duì)本系列文章感興趣,歡迎關(guān)注訂閱這里這系列文章講什么本系列文章主要介紹如何用這一當(dāng)前流行的構(gòu)建工具來(lái)設(shè)計(jì)一個(gè)多頁(yè)應(yīng)用的架構(gòu)。 本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。原文地址:https://segmentfault.com/a/1190000006843916如果您對(duì)本系列文章...
摘要:在上一篇文章多頁(yè)應(yīng)用架構(gòu)系列二配置常用部分有哪些中,我介紹了如何配置多頁(yè)應(yīng)用的入口,然而,如果僅僅如此操作,帶來(lái)的后果就是,打包生成出來(lái)的每一個(gè)入口文件都會(huì)完整包含所有代碼。的初始化常用參數(shù)有哪些,給這個(gè)包含公共代碼的命個(gè)名唯一標(biāo)識(shí)。 本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。原文地址:https://segmentfault.com/a/1190...
摘要:本文首發(fā)于的技術(shù)博客實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。如果你使用了,或類(lèi)似的,那么,通過(guò)編譯前后的代碼相差就很大了,這會(huì)造成兩個(gè)問(wèn)題以為例把你的代碼轉(zhuǎn)成什么樣你自己是無(wú)法控制的,這往往導(dǎo)致無(wú)法通過(guò)的審查。 本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。原文地址:https://segmentfault.com/a/1190000007030775如果您...
摘要:徹底分離源文件目錄和生成文件目錄使用生成出來(lái)的頁(yè)面可以很安心地跟打包好的其它資源放到一起,相對(duì)于另起一個(gè)目錄專(zhuān)門(mén)存放頁(yè)面文件來(lái)說(shuō),整個(gè)文件目錄結(jié)構(gòu)更加合理如何利用生成頁(yè)面生成頁(yè)面主要是通過(guò)來(lái)實(shí)現(xiàn)的,下面來(lái)介紹如何實(shí)現(xiàn)。 本文首發(fā)于Array_Huang的技術(shù)博客——實(shí)用至上,非經(jīng)作者同意,請(qǐng)勿轉(zhuǎn)載。原文地址:https://segmentfault.com/a/119000000712...
閱讀 1476·2021-11-16 11:44
閱讀 3296·2021-09-29 09:43
閱讀 627·2019-08-30 10:52
閱讀 946·2019-08-29 11:01
閱讀 3264·2019-08-26 11:47
閱讀 2897·2019-08-23 12:18
閱讀 1369·2019-08-22 17:04
閱讀 2055·2019-08-21 17:04