注:本文使用的 react-router 版本為 2.8.1
10多年的營(yíng)口網(wǎng)站建設(shè)經(jīng)驗(yàn),針對(duì)設(shè)計(jì)、前端、開(kāi)發(fā)、售后、文案、推廣等六對(duì)一服務(wù),響應(yīng)快,48小時(shí)及時(shí)工作處理。成都全網(wǎng)營(yíng)銷(xiāo)的優(yōu)勢(shì)是能夠根據(jù)用戶(hù)設(shè)備顯示端的尺寸不同,自動(dòng)調(diào)整營(yíng)口建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無(wú)論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計(jì),從而大程度地提升瀏覽體驗(yàn)。成都創(chuàng)新互聯(lián)從事“營(yíng)口網(wǎng)站設(shè)計(jì)”,“營(yíng)口網(wǎng)站推廣”以來(lái),每個(gè)客戶(hù)項(xiàng)目都認(rèn)真落實(shí)執(zhí)行。
React Router 是一個(gè)非常出色的路由解決方案,同時(shí)也非常容易上手。但是當(dāng)網(wǎng)站規(guī)模越來(lái)越大的時(shí)候,首先出現(xiàn)的問(wèn)題是 Javascript 文件變得巨大,這導(dǎo)致首頁(yè)渲染的時(shí)間讓人難以忍受。實(shí)際上程序應(yīng)當(dāng)只加載當(dāng)前渲染頁(yè)所需的 JavaScript,也就是大家說(shuō)的“代碼分拆" — 將所有的代碼分拆成多個(gè)小包,在用戶(hù)瀏覽過(guò)程中按需加載。
所得到的效果是:
以前是這樣(23333,我真不是故意的。。)
現(xiàn)在是這樣:
實(shí)際上就是將一個(gè)大 javascript 文件拆分成了若干個(gè) chunk file。
下面是改造過(guò)程
Webpack 配置
首先在 webpack.config.js
的 output
內(nèi)加上 chunkFilename
output: { path: path.join(__dirname, '/../dist/assets'), filename: 'app.js', publicPath: defaultSettings.publicPath, // 添加 chunkFilename chunkFilename: '[name].[chunkhash:5].chunk.js', },
name
是在代碼里為創(chuàng)建的 chunk 指定的名字,如果代碼中沒(méi)指定則 webpack 默認(rèn)分配 id 作為 name。
chunkhash
是文件的 hash 碼,這里只使用前五位。
添加首頁(yè)
以前你的路由大概應(yīng)該是這樣的:(作為需要按需加載的大型應(yīng)用,路由肯定是相當(dāng)復(fù)雜,這里只列舉部分路由舉例)
ReactDOM.render( ({/* 主頁(yè) */} ), document.getElementById('app') );{/* 默認(rèn) */} {/* baidu */} {/* 404 */} {/* 其他重定向到 404 */}
按需加載之后,我們需要讓路由動(dòng)態(tài)加載組件,需要將 component
換成 getComponent
。首先將路由拆出來(lái)(因?yàn)槁酚升嫶笾笕繉?xiě)在一起會(huì)很難看),創(chuàng)建一個(gè)根路由 rootRoute:
const rootRoute = { path: '/', indexRoute: { getComponent(nextState, cb) { require.ensure([], (require) => { cb(null, require('components/layer/HomePage')) }, 'HomePage') }, }, getComponent(nextState, cb) { require.ensure([], (require) => { cb(null, require('components/Main')) }, 'Main') }, childRoutes: [ require('./routes/baidu'), require('./routes/404'), require('./routes/redirect') ] } ReactDOM.render( (), document.getElementById('app') );
history 不變,在 Router 中添加 routes 屬性,將創(chuàng)建的路由傳遞進(jìn)去。
這里有四個(gè)屬性:
path
將匹配的路由,也就是以前的 path。
getComponent
對(duì)應(yīng)于以前的 component 屬性,但是這個(gè)方法是異步的,也就是當(dāng)路由匹配時(shí),才會(huì)調(diào)用這個(gè)方法。
這里面有個(gè) require.ensure 方法
require.ensure(dependencies, callback, chunkName)
這是 webpack 提供的方法,這也是按需加載的核心方法。第一個(gè)參數(shù)是依賴(lài),第二個(gè)是回調(diào)函數(shù),第三個(gè)就是上面提到的 chunkName,用來(lái)指定這個(gè) chunk file 的 name。
如果需要返回多個(gè)子組件,則使用 getComponents 方法,將多個(gè)組件作為一個(gè)對(duì)象的屬性通過(guò) cb 返回出去即可。這個(gè)在官方示例也有,但是我們這里并不需要,而且根組件是不能返回多個(gè)子組件的,所以使用 getComponent。
indexRoute
用來(lái)設(shè)置主頁(yè),對(duì)應(yīng)于以前的
注意這里的 indexRoute 寫(xiě)法, 這是個(gè)對(duì)象,在對(duì)象里面使用 getComponent。
childRoutes
這里面放置的就是子路由的配置,對(duì)應(yīng)于以前的子路由們。我們將以前的 /baidu、/404 和 * 都拆了出來(lái),接下來(lái)將分別為他們創(chuàng)建路由配置。
路由控制
上面的childRoutes
里面,我們 require
了三個(gè)子路由,在目錄下創(chuàng)建 routes
目錄,將這三個(gè)路由放置進(jìn)去。
routes/ ├── 404 │ └── index.js ├── baidu │ ├── index.js │ └── routes │ ├── frequency │ │ └── index.js │ └── result │ └── index.js └── redirect └── index.js
和 rootRoute 類(lèi)似,里面的每個(gè) index.js 都是一個(gè)路由對(duì)象:
/404/index.js
module.exports = { path: '404', getComponent(nextState, cb) { require.ensure([], (require) => { cb(null, require('components/layer/NotFoundPage')) }, 'NotFoundPage') } }
/baidu/index.js
module.exports = { path: 'baidu', getChildRoutes(partialNextState, cb) { require.ensure([], (require) => { cb(null, [ require('./routes/result'), require('./routes/frequency') ]) }) }, getComponent(nextState, cb) { require.ensure([], (require) => { cb(null, require('components/layer/BaiduPage')) }, 'BaiduPage') } }
/baidu/routes/frequency/index.js
module.exports = { path: 'frequency', getComponent(nextState, cb) { require.ensure([], (require) => { cb(null, require('components/layer/BaiduFrequencyPage')) }, 'BaiduFrequencyPage') } }
舉這幾個(gè)例子應(yīng)該就差不多了,其他都是一樣的,稍微有點(diǎn)特別的是 redirect。
設(shè)置 Redirect
之前我們?cè)诟酚上率沁@么設(shè)置重定向的:
{/* home */} {/* 如果都不匹配,重定向到 404 */}
當(dāng)改寫(xiě)之后,我們需要把這個(gè)重定向的路由單獨(dú)拆出來(lái),也就是 * 這個(gè)路由,我們上面已經(jīng)為他創(chuàng)建了一個(gè) redirect 目錄。這里使用到 onEnter 方法,然后在這個(gè)方法里改變路由狀態(tài),調(diào)到另外的路由,實(shí)現(xiàn) redirect :
/redirect/index.js
module.exports = { path: '*', onEnter: (_, replaceState) => replaceState(null, "/404") }
The root route must render a single element
跟著官方示例和上面碼出來(lái)之后,可能頁(yè)面并沒(méi)有渲染出來(lái),而是報(bào) The root route must render a single element 這個(gè)異常,這是因?yàn)?code> module.exports 和 ES6 里的 export default
有區(qū)別。
如果你是使用 es6 的寫(xiě)法,也就是你的組件都是通過(guò) export default 導(dǎo)出的,那么在 getComponent 方法里面需要加入 .default。
getComponent(nextState, cb) { require.ensure([], (require) => { // 在后面加 .default cb(null, require('components/layer/ReportPage')).default }, 'ReportPage') }
如果你是使用 CommonJS 的寫(xiě)法,也就是通過(guò) module.exports
導(dǎo)出的,那就無(wú)須加 .default
了。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。