詳解react-router如何實(shí)現(xiàn)按需加載
注:本文使用的 react-router 版本為 2.8.1
React Router 是一個(gè)非常出色的路由解決方案,同時(shí)也非常容易上手。但是當(dāng)網(wǎng)站規(guī)模越來越大的時(shí)候,首先出現(xiàn)的問題是 Javascript 文件變得巨大,這導(dǎo)致首頁渲染的時(shí)間讓人難以忍受。實(shí)際上程序應(yīng)當(dāng)只加載當(dāng)前渲染頁所需的 JavaScript,也就是大家說的“代碼分拆" — 將所有的代碼分拆成多個(gè)小包,在用戶瀏覽過程中按需加載。
所得到的效果是:
以前是這樣(23333,我真不是故意的。。)
![]()
現(xiàn)在是這樣:

實(shí)際上就是將一個(gè)大 javascript 文件拆分成了若干個(gè) chunk file。
下面是改造過程
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 指定的名字,如果代碼中沒指定則 webpack 默認(rèn)分配 id 作為 name。
chunkhash 是文件的 hash 碼,這里只使用前五位。
添加首頁
以前你的路由大概應(yīng)該是這樣的:(作為需要按需加載的大型應(yīng)用,路由肯定是相當(dāng)復(fù)雜,這里只列舉部分路由舉例)
ReactDOM.render(
(
<Router history={browserHistory}>
{/* 主頁 */}
<Route path="/" component={App}>
{/* 默認(rèn) */}
<IndexRoute component={HomePage} />
{/* baidu */}
<Route path="/baidu" component={BaiduPage}>
<Route path="result" component={BaiduResultPage} />
<Route path="frequency" component={BaiduFrequencyPage} />
</Route>
{/* 404 */}
<Route path='/404' component={NotFoundPage} />
{/* 其他重定向到 404 */}
<Redirect from='*' to='/404' />
</Route>
</Router>
), document.getElementById('app')
);
按需加載之后,我們需要讓路由動(dòng)態(tài)加載組件,需要將 component 換成 getComponent。首先將路由拆出來(因?yàn)槁酚升嫶笾笕繉懺谝黄饡?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(
(
<Router
history={browserHistory}
routes={rootRoute}
/>
), 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ù)是依賴,第二個(gè)是回調(diào)函數(shù),第三個(gè)就是上面提到的 chunkName,用來指定這個(gè) chunk file 的 name。
如果需要返回多個(gè)子組件,則使用 getComponents 方法,將多個(gè)組件作為一個(gè)對(duì)象的屬性通過 cb 返回出去即可。這個(gè)在官方示例也有,但是我們這里并不需要,而且根組件是不能返回多個(gè)子組件的,所以使用 getComponent。
indexRoute
用來設(shè)置主頁,對(duì)應(yīng)于以前的 <IndexRoute>。
注意這里的 indexRoute 寫法, 這是個(gè)對(duì)象,在對(duì)象里面使用 getComponent。
childRoutes
這里面放置的就是子路由的配置,對(duì)應(yīng)于以前的子路由們。我們將以前的 /baidu、/404 和 * 都拆了出來,接下來將分別為他們創(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 類似,里面的每個(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è)置重定向的:
<Router history={browserHistory}>
<Route path="/" component={App}>
{/* home */}
<IndexRoute component={HomePage} />
<Route path="/baidu" component={BaiduPage}>
<Route path="result" component={BaiduResultPage} />
<Route path="frequency" component={BaiduFrequencyPage} />
</Route>
<Route path='/404' component={NotFoundPage} />
{/* 如果都不匹配,重定向到 404 */}
<Redirect from='*' to='/404' />
</Route>
</Router>
當(dāng)改寫之后,我們需要把這個(gè)重定向的路由單獨(dú)拆出來,也就是 * 這個(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
跟著官方示例和上面碼出來之后,可能頁面并沒有渲染出來,而是報(bào) The root route must render a single element 這個(gè)異常,這是因?yàn)?code> module.exports 和 ES6 里的 export default 有區(qū)別。
如果你是使用 es6 的寫法,也就是你的組件都是通過 export default 導(dǎo)出的,那么在 getComponent 方法里面需要加入 .default。
getComponent(nextState, cb) {
require.ensure([], (require) => {
// 在后面加 .default
cb(null, require('components/layer/ReportPage')).default
}, 'ReportPage')
}
如果你是使用 CommonJS 的寫法,也就是通過 module.exports 導(dǎo)出的,那就無須加 .default 了。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
antd之RangePicker設(shè)置默認(rèn)值方式
這篇文章主要介紹了antd之RangePicker設(shè)置默認(rèn)值方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
教你如何實(shí)現(xiàn)在react項(xiàng)目中嵌入Blazor
這篇文章主要介紹了如何實(shí)現(xiàn)在react現(xiàn)有項(xiàng)目中嵌入Blazor,通過這個(gè)案例我們可以知道 blazor也可以像react那樣嵌入在任何的現(xiàn)有項(xiàng)目中,并且使用方便,需要的朋友可以參考下2023-01-01
react 頁面加載完成后自動(dòng)執(zhí)行標(biāo)簽的點(diǎn)擊事件的兩種操作方法
這篇文章主要介紹了react 頁面加載完成后自動(dòng)執(zhí)行標(biāo)簽的點(diǎn)擊事件,本文給大家分享兩種操作方法結(jié)合示例代碼給大家講解的非常詳細(xì),需要的朋友可以參考下2022-12-12
使用React?SSR寫Demo一學(xué)就會(huì)
這篇文章主要為大家介紹了使用React?SSR寫Demo實(shí)現(xiàn)教程示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
react-three-fiber實(shí)現(xiàn)炫酷3D粒子效果首頁
這篇文章主要介紹了react-three-fiber實(shí)現(xiàn)3D粒子效果,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08

