本文主要介紹 UmiJS 的預(yù)渲染功能。
旺蒼網(wǎng)站建設(shè)公司成都創(chuàng)新互聯(lián),旺蒼網(wǎng)站設(shè)計(jì)制作,有大型網(wǎng)站制作公司豐富經(jīng)驗(yàn)。已為旺蒼上1000家提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\成都外貿(mào)網(wǎng)站制作要多少錢(qián),請(qǐng)找那個(gè)售后服務(wù)好的旺蒼做網(wǎng)站的公司定做!
服務(wù)端渲染(Server-Side Rendering),是指由 服務(wù)端 完成頁(yè)面的 HTML 結(jié)構(gòu)拼接的頁(yè)面處理技術(shù),發(fā)送到瀏覽器,然后為其綁定狀態(tài)與事件,成為完全可交互頁(yè)面的過(guò)程。
服務(wù)端渲染,首先得有后端服務(wù)器(一般是 Node.js)才可以使用,而沒(méi)有后端服務(wù)器的情況下,可以使用 預(yù)渲染 。
預(yù)渲染與服務(wù)端渲染唯一的不同點(diǎn)在于 渲染時(shí)機(jī) ,服務(wù)端渲染的時(shí)機(jī)是在用戶(hù)訪問(wèn)時(shí)執(zhí)行渲染(即實(shí)時(shí)渲染,數(shù)據(jù)一般是最新的),預(yù)渲染的時(shí)機(jī)是在項(xiàng)目構(gòu)建時(shí),當(dāng)用戶(hù)訪問(wèn)時(shí),數(shù)據(jù)不一定是最新的( 如果數(shù)據(jù)沒(méi)有實(shí)時(shí)性,可以直接考慮預(yù)渲染 )。
預(yù)渲染在構(gòu)建時(shí)執(zhí)行渲染,將渲染后的 HTML 片段生成靜態(tài) HTML 文件。無(wú)需使用 web 服務(wù)器實(shí)時(shí)動(dòng)態(tài)編譯 HTML,適用于 靜態(tài)站點(diǎn)生成 。
Umi3 在 SSR 上做了大量?jī)?yōu)化及開(kāi)發(fā)體驗(yàn)的提升,具有以下特性:
默認(rèn)情況下,服務(wù)端渲染時(shí)關(guān)閉的,可通過(guò)配置開(kāi)啟:
服務(wù)端渲染的數(shù)據(jù)獲取方式與 SPA(單頁(yè)應(yīng)用) 有所不同,為了讓客戶(hù)端和服務(wù)端都能獲取到同一份數(shù)據(jù),Umi 提供了頁(yè)面級(jí)數(shù)據(jù)的預(yù)獲取。
每個(gè)頁(yè)面可能有單獨(dú)的數(shù)據(jù)預(yù)獲取邏輯,這里我們會(huì)獲取頁(yè)面組件上的 getInitialProps 靜態(tài)方法,執(zhí)行后將結(jié)果注入到該頁(yè)面組件的 props 中,如:
getInitialProps 有幾個(gè)固定參數(shù):
為了結(jié)合數(shù)據(jù)流框架,我們提供了 modifyGetInitialPropsCtx 方法,由插件或應(yīng)用來(lái)擴(kuò)展 ctx 參數(shù),以 dva 為例:
然后在頁(yè)面中,可以獲取到 store :
同時(shí)也可以在自身應(yīng)用中進(jìn)行擴(kuò)展:
同時(shí)可以使用 getInitialPropsCtx 將服務(wù)端參數(shù)擴(kuò)展到 ctx 中,例如:
在使用的時(shí)候,就有 req 對(duì)象,不過(guò)需要注意的是,只在服務(wù)端執(zhí)行時(shí)才有此參數(shù):
則在執(zhí)行 getInitialProps 方法時(shí),除了以上兩個(gè)固定參數(shù)外,還會(huì)獲取到 title 和 store 參數(shù)。
關(guān)于 getInitialProps 執(zhí)行邏輯和時(shí)機(jī),需要注意:
執(zhí)行 umi build ,除了正常的 umi.js 外,會(huì)多一個(gè)服務(wù)端文件: umi.server.js (相當(dāng)于服務(wù)端入口文件)。然后在后端框架中,引用該文件:
render 方法參數(shù)和返回值如下:
完美兼容客戶(hù)端動(dòng)態(tài)加載,配置如下:
@umijs/preset-react 插件集中已內(nèi)置對(duì)標(biāo)題的渲染,通過(guò)以下步驟使用:
@umijs/preset-react 插件集中已內(nèi)置 dva
這時(shí)候 getInitialProps(ctx) 中的 ctx 就會(huì)有 store 屬性,可執(zhí)行 dispatch ,并返回初始化數(shù)據(jù)。
Umi 同時(shí)支持對(duì)服務(wù)端和客戶(hù)端包大小的分析
需要準(zhǔn)備的材料分別有:電腦、html編輯器、瀏覽器。
1、首先,打開(kāi)html編輯器,新建html文件,例如:index.html。
2、在index.html中的script標(biāo)簽,輸入js代碼:var a = 'input type="text" value="test" /';document.body.innerText = a;。
3、瀏覽器運(yùn)行index.html頁(yè)面,此時(shí)html代碼被當(dāng)成字符串渲染到了頁(yè)面上。
對(duì)于以下視圖樹(shù)
如何實(shí)現(xiàn)視圖樹(shù)中的三目操作符呢
React 的做法是生成虛擬樹(shù),與舊樹(shù)diff
在Vanilla js中怎么做?
視圖樹(shù)中的三目操作符實(shí)際上表達(dá)了兩種操作:
刪除節(jié)點(diǎn)好刪,直接刪了就是
添加節(jié)點(diǎn)呢,又分為兩步
這插入到指定位置就麻煩了,需要知道要插入到哪個(gè)位置呢?信息不足啊。
有如下方案:
render是渲染的意思。
res.render(file,option)是express中專(zhuān)門(mén)渲染視圖用的,首先你要在你的app.js或者index.js中設(shè)置一下渲染引擎,比如html,jade,handlebars(我自己使用的),mustache等。然后將視圖模板的文件位置放入file,將傳入的模板數(shù)據(jù)放入option對(duì)象中,模板引擎就能自己渲染出視圖。給你推薦一個(gè)npm模塊,express-handlebars,能很快搭建一個(gè)項(xiàng)目,你實(shí)踐過(guò),就能明白res.render.
而渲染,就是這樣一個(gè)過(guò)程,瀏覽器根據(jù)頁(yè)面的html代碼、css定義、javascript腳本的操作,在瀏覽器上按照一定的規(guī)范(傳說(shuō)中的 DOCTYPE )顯示出相應(yīng)的內(nèi)容。
舉個(gè)最簡(jiǎn)單的例子:
b這個(gè)是粗體顯示/b
瀏覽器根據(jù)上面的代碼在瀏覽器上面顯示出粗體的文字,這個(gè)過(guò)程就叫渲染
查看一下效果,應(yīng)該很容易得到結(jié)果
有時(shí)我們可能需要重復(fù)一個(gè)包含多個(gè)節(jié)點(diǎn)的塊,這時(shí)可以用 template 標(biāo)簽包裹這個(gè)塊。這里的 template 標(biāo)簽只起到語(yǔ)義上的包裹作用,其本身不會(huì)被渲染出來(lái)。例如:
簡(jiǎn)單值 (primitive value) 即字符串、數(shù)字、boolean 等并非對(duì)象的值。對(duì)于包含簡(jiǎn)單值的數(shù)組,你可用 $value 直接訪問(wèn)值:
有時(shí)我們可能想要更明確地訪問(wèn)當(dāng)前作用域的變量而不是隱式地回退到父作用域。你可以通過(guò)提供一個(gè)參數(shù)給 v-repeat 指令并用它作為將被迭代項(xiàng)的別名:
Vue.js 內(nèi)部對(duì)被觀察數(shù)組的變異方法 (mutating methods,包括 push(), pop(), shift(), unshift(), splice(), sort() 和 reverse()) 進(jìn)行了攔截,因此調(diào)用這些方法也將自動(dòng)觸發(fā)視圖更新。
下面是一個(gè)演示的例子,點(diǎn)擊按鈕的時(shí)候數(shù)據(jù)項(xiàng)會(huì)被移除
Vue.js 給被觀察數(shù)組添加了兩個(gè)便捷方法:$set() 和 $remove() 。
你應(yīng)該避免直接通過(guò)索引來(lái)設(shè)置數(shù)據(jù)綁定數(shù)組中的元素,比如 demo.items[0] = {},因?yàn)檫@些改動(dòng)是無(wú)法被 Vue.js 偵測(cè)到的。你應(yīng)該使用擴(kuò)展的 $set() 方法:
$remove() 只是 splice()方法的語(yǔ)法糖。它將移除給定索引處的元素。當(dāng)參數(shù)不是數(shù)值時(shí),$remove() 將在數(shù)組中搜索該值并刪除第一個(gè)發(fā)現(xiàn)的對(duì)應(yīng)元素。
當(dāng)你使用非變異方法,比如filter(), concat() 或 slice(),返回的數(shù)組將是一個(gè)不同的實(shí)例。在此情況下,你可以用新數(shù)組替換舊的數(shù)組:
你可能會(huì)認(rèn)為這將導(dǎo)致整個(gè)列表的 DOM 被銷(xiāo)毀并重新渲染。但別擔(dān)心,Vue.js 能夠識(shí)別一個(gè)數(shù)組元素是否已有關(guān)聯(lián)的 Vue 實(shí)例, 并盡可能地進(jìn)行有效復(fù)用。
在某些情況下,你可能需要以全新的對(duì)象(比如 API 調(diào)用所返回的對(duì)象)去替換數(shù)組。如果你的每一個(gè)數(shù)據(jù)對(duì)象都有一個(gè)唯一的 id 屬性,那么你可以使用 track-by 特性參數(shù)給 Vue.js 一個(gè)提示,這樣它就可以復(fù)用已有的具有相同 id 的 Vue 實(shí)例和 DOM 節(jié)點(diǎn)。
例如:你的數(shù)據(jù)是這個(gè)樣子的
那么你可以像這樣給出提示:
在替換 items 數(shù)組時(shí),Vue.js 如果碰到一個(gè)有 _uid: '88f869d' 的新對(duì)象,它就會(huì)知道可以直接復(fù)用有同樣 _uid 的已有實(shí)例。 在使用全新數(shù)據(jù)重新渲染大型 v-repeat 列表時(shí),合理使用 track-by 能極大地提升性能。
你也可以使用 v-repeat 遍歷一個(gè)對(duì)象的所有屬性。每個(gè)重復(fù)的實(shí)例會(huì)有一個(gè)特殊的屬性 $key。對(duì)于簡(jiǎn)單值,你也可以象訪問(wèn)數(shù)組中的簡(jiǎn)單值那樣使用 $value 屬性。
在 ECMAScript 5 中,對(duì)于給對(duì)象添加一個(gè)新屬性,或是從對(duì)象中刪除一個(gè)屬性時(shí),沒(méi)有機(jī)制可以檢測(cè)到這兩種情況。針對(duì)這個(gè)問(wèn)題,Vue.js 中的被觀察對(duì)象會(huì)被添加三個(gè)擴(kuò)展方法: $add(key, value), $set(key, value) 和 $delete(key)。這些方法可以被用于在添加 / 刪除觀察對(duì)象的屬性時(shí)觸發(fā)對(duì)應(yīng)的視圖更新。方法 $add 和 $set 的不同之處在于當(dāng)指定的鍵已經(jīng)在對(duì)象中存在時(shí) $add 將提前返回,所以調(diào)用 obj.$add(key) 并不會(huì)以 undefined 覆蓋已有的值。
v-repeat 也可以接受一個(gè)整數(shù)。在這種情況下,它將重復(fù)顯示一個(gè)模板多次。 下面的例子將迭代3次。
有時(shí)候我們只需要顯示一個(gè)數(shù)組的過(guò)濾或排序過(guò)的版本,而不需要實(shí)際改變或重置原始數(shù)據(jù)。Vue 提供了兩個(gè)內(nèi)置的過(guò)濾器來(lái)簡(jiǎn)化此類(lèi)需求: filterBy 和 orderBy。