詳解react-router如何實(shí)現(xiàn)按需加載
注:本文使用的 react-router 版本為 2.8.1
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è)小包,在用戶瀏覽過(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( ( <Router history={browserHistory}> {/* 主頁(yè) */} <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
。首先將路由拆出來(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( ( <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,用來(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>。
注意這里的 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 類似,里面的每個(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)改寫(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í)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
React進(jìn)階學(xué)習(xí)之組件的解耦之道
這篇文章主要給大家介紹了關(guān)于React進(jìn)階之組件的解耦之道,文中通過(guò)詳細(xì)的示例代碼給大家介紹了組件分割與解耦的方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面跟著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-08-08React?Native集成支付寶支付的實(shí)現(xiàn)方法
這篇文章主要介紹了React?Native集成支付寶支付的實(shí)現(xiàn)現(xiàn),ativeModules是JS代碼調(diào)用原生模塊的橋梁。所以,我們只需要在原生工程中集成支付寶和微信支付的sdk,然后使用NativeModules調(diào)用即可,需要的朋友可以參考下2022-02-02淺談React + Webpack 構(gòu)建打包優(yōu)化
本篇文章主要介紹了淺談React + Webpack 構(gòu)建打包優(yōu)化,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter
這篇文章主要介紹了詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-12-12