亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

詳解react-router如何實現按需加載

2019-11-19 16:19:50
字體:
來源:轉載
供稿:網友

注:本文使用的 react-router 版本為 2.8.1

React Router 是一個非常出色的路由解決方案,同時也非常容易上手。但是當網站規模越來越大的時候,首先出現的問題是 Javascript 文件變得巨大,這導致首頁渲染的時間讓人難以忍受。實際上程序應當只加載當前渲染頁所需的 JavaScript,也就是大家說的“代碼分拆" ― 將所有的代碼分拆成多個小包,在用戶瀏覽過程中按需加載。

所得到的效果是:

以前是這樣(23333,我真不是故意的。。)

現在是這樣:

實際上就是將一個大 javascript 文件拆分成了若干個 chunk file。

下面是改造過程

Webpack 配置

首先在 webpack.config.js output 內加上 chunkFilename

output: {  path: path.join(__dirname, '/../dist/assets'),  filename: 'app.js',  publicPath: defaultSettings.publicPath,  // 添加 chunkFilename  chunkFilename: '[name].[chunkhash:5].chunk.js',},

name 是在代碼里為創建的 chunk 指定的名字,如果代碼中沒指定則 webpack 默認分配 id 作為 name。

chunkhash 是文件的 hash 碼,這里只使用前五位。

添加首頁

以前你的路由大概應該是這樣的:(作為需要按需加載的大型應用,路由肯定是相當復雜,這里只列舉部分路由舉例)

ReactDOM.render( (  <Router history={browserHistory}>   {/* 主頁 */}   <Route path="/" component={App}>    {/* 默認 */}    <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'));

按需加載之后,我們需要讓路由動態加載組件,需要將 component 換成 getComponent。首先將路由拆出來(因為路由龐大之后全部寫在一起會很難看),創建一個根路由 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 屬性,將創建的路由傳遞進去。

這里有四個屬性:

path

將匹配的路由,也就是以前的 path。

getComponent

對應于以前的 component 屬性,但是這個方法是異步的,也就是當路由匹配時,才會調用這個方法。

這里面有個 require.ensure 方法

require.ensure(dependencies, callback, chunkName)

這是 webpack 提供的方法,這也是按需加載的核心方法。第一個參數是依賴,第二個是回調函數,第三個就是上面提到的 chunkName,用來指定這個 chunk file 的 name。

如果需要返回多個子組件,則使用 getComponents 方法,將多個組件作為一個對象的屬性通過 cb 返回出去即可。這個在官方示例也有,但是我們這里并不需要,而且根組件是不能返回多個子組件的,所以使用 getComponent。

indexRoute

用來設置主頁,對應于以前的 <IndexRoute>。

注意這里的 indexRoute 寫法, 這是個對象,在對象里面使用 getComponent。

childRoutes

這里面放置的就是子路由的配置,對應于以前的子路由們。我們將以前的 /baidu、/404 和 * 都拆了出來,接下來將分別為他們創建路由配置。

路由控制

上面的childRoutes 里面,我們 require 了三個子路由,在目錄下創建 routes 目錄,將這三個路由放置進去。

routes/├── 404│  └── index.js├── baidu│  ├── index.js│  └── routes│    ├── frequency│    │  └── index.js│    └── result│      └── index.js└── redirect  └── index.js

和 rootRoute 類似,里面的每個 index.js 都是一個路由對象:

/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') }}

舉這幾個例子應該就差不多了,其他都是一樣的,稍微有點特別的是 redirect。

設置 Redirect

之前我們在根路由下是這么設置重定向的:

<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>

當改寫之后,我們需要把這個重定向的路由單獨拆出來,也就是 * 這個路由,我們上面已經為他創建了一個 redirect 目錄。這里使用到 onEnter 方法,然后在這個方法里改變路由狀態,調到另外的路由,實現 redirect :

/redirect/index.js

module.exports = { path: '*', onEnter: (_, replaceState) => replaceState(null, "/404")}

The root route must render a single element

跟著官方示例和上面碼出來之后,可能頁面并沒有渲染出來,而是報 The root route must render a single element 這個異常,這是因為 module.exports 和 ES6 里的 export default 有區別。

如果你是使用 es6 的寫法,也就是你的組件都是通過 export default 導出的,那么在 getComponent 方法里面需要加入 .default。

getComponent(nextState, cb) {  require.ensure([], (require) => {   // 在后面加 .default   cb(null, require('components/layer/ReportPage')).default  }, 'ReportPage')}

如果你是使用 CommonJS 的寫法,也就是通過 module.exports 導出的,那就無須加 .default 了。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产福利成人在线| 亚洲免费电影在线观看| 97成人精品视频在线观看| 国产视频精品免费播放| 久久精品免费电影| 国产福利精品视频| 日韩麻豆第一页| 亚洲深夜福利在线| 91精品国产高清| 国产一区二区三区毛片| 欧美中文在线字幕| 91久热免费在线视频| 国产中文欧美精品| 另类图片亚洲另类| 欧美激情乱人伦| 亚洲精品一区二区久| 日韩欧美成人网| 97视频网站入口| 欧美区在线播放| 亚洲欧洲第一视频| 欧美视频在线观看免费网址| 亚洲va国产va天堂va久久| 中文字幕在线成人| 日韩国产精品亚洲а∨天堂免| 91国自产精品中文字幕亚洲| 国产亚洲综合久久| 亚洲欧美国产va在线影院| 欧美性极品xxxx娇小| 亚洲人精选亚洲人成在线| www.99久久热国产日韩欧美.com| 日韩中文av在线| 午夜精品久久久久久久99热浪潮| 久久色精品视频| 日韩亚洲综合在线| 久久免费精品日本久久中文字幕| 亚洲人成自拍网站| 国产精品日韩在线| 欧美日韩美女视频| 亚洲电影免费观看高清完整版| 欧美性猛交xxxx乱大交3| 国产精品亚洲视频在线观看| 国产日韩欧美日韩大片| 日韩激情片免费| 亚洲精品小视频| 68精品久久久久久欧美| 97国产精品免费视频| 欧美日韩第一视频| 欧美日韩一区二区三区在线免费观看| 欧美极品少妇xxxxⅹ免费视频| 久久久精品日本| 91av在线影院| 日韩在线一区二区三区免费视频| 国产精品视频自在线| 久久久久久久久电影| 亚洲一区二区免费在线| 国产精品一区二区三区毛片淫片| 国自产精品手机在线观看视频| 精品亚洲永久免费精品| 国产日韩在线看片| 中文字幕欧美视频在线| 亚洲欧美国产精品va在线观看| 久久久久亚洲精品国产| 中文字幕av一区二区| 欧美综合一区第一页| 国产一区二区三区在线观看网站| 国产精品吹潮在线观看| 成人中心免费视频| 国产精品一区二区电影| 亚洲精品在线不卡| 国产日本欧美在线观看| 欧美夫妻性生活视频| 久久6免费高清热精品| 日本道色综合久久影院| 国内精品伊人久久| 日韩欧美高清视频| 亚洲一二三在线| 国产精品日韩欧美| 在线观看日韩www视频免费| 91免费看片网站| 成人深夜直播免费观看| 成人国产精品一区| 91国内在线视频| 国产综合色香蕉精品| 欧美激情在线视频二区| 久久精品国产精品亚洲| 国产欧美精品xxxx另类| 欧美一级视频在线观看| 中文字幕精品影院| 在线成人中文字幕| 91国产一区在线| 亚洲乱码一区二区| 亚洲国产成人精品久久| 国产mv久久久| 国产精品免费视频xxxx| 欧美二区在线播放| 日韩中文字幕在线视频| 尤物yw午夜国产精品视频| 欧美激情在线有限公司| 高清欧美电影在线| 欧美色欧美亚洲高清在线视频| 国产乱肥老妇国产一区二| 国产一区二区色| 亚洲欧美另类国产| 亚洲另类激情图| 美女福利精品视频| 亚洲人精选亚洲人成在线| 欧美性猛交xxxx| 亚洲欧洲中文天堂| 日韩hd视频在线观看| 国产欧美日韩亚洲精品| 日韩欧美国产网站| 日本精品久久久久久久| 欧美黄色成人网| 伊人青青综合网站| 91久久久久久久久| 亚洲精美色品网站| 欧美黑人巨大xxx极品| 91av在线播放| 成人伊人精品色xxxx视频| 亚洲欧美在线看| 欧美片一区二区三区| 亚洲深夜福利网站| 久久精视频免费在线久久完整在线看| 日韩av色综合| 在线免费观看羞羞视频一区二区| 亚洲精品动漫久久久久| 亚洲风情亚aⅴ在线发布| 日韩欧美高清视频| 亚洲在线免费观看| 最近2019年日本中文免费字幕| 欧美成人精品影院| 国产精品久久久一区| 中文字幕欧美国内| 久久精品国产久精国产思思| 亚洲色无码播放| 亚洲影视中文字幕| 欧美激情一二区| 国产精品久久久久91| 丝袜亚洲另类欧美重口| 国产日韩专区在线| 亚洲综合最新在线| 亚洲精品国产精品乱码不99按摩| 亚洲精品电影久久久| 91久久久久久久久久久久久| 岛国精品视频在线播放| 午夜精品99久久免费| 精品国产鲁一鲁一区二区张丽| 亚洲精品一区中文字幕乱码| 日韩中文字幕av| 久久久久久亚洲精品中文字幕| 美女999久久久精品视频| 亚洲精品国产综合久久| 日韩在线免费高清视频| 国产一区二区三区在线| 国产一区视频在线播放| 亚洲欧洲免费视频| 欧美日韩另类在线| 亚洲成av人片在线观看香蕉| 日韩国产欧美区| 一本色道久久88综合亚洲精品ⅰ| 亚洲国产精品999| 91高清视频免费观看| 69影院欧美专区视频| 国产精品亚洲аv天堂网|