亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳解react-router如何實(shí)現(xiàn)按需加載

 更新時(shí)間:2017年06月15日 14:23:16   作者:時(shí)間被海綿吃了  
本篇文章主要介紹了react-router如何實(shí)現(xiàn)按需加載,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

注:本文使用的 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)文章

  • 2個(gè)奇怪的react寫(xiě)法

    2個(gè)奇怪的react寫(xiě)法

    大家好,我卡頌。雖然React官網(wǎng)用大量篇幅介紹最佳實(shí)踐,但因JSX語(yǔ)法的靈活性,所以總是會(huì)出現(xiàn)奇奇怪怪的React寫(xiě)法。本文介紹2種奇怪(但在某些場(chǎng)景下有意義)的React寫(xiě)法。也歡迎大家在評(píng)論區(qū)討論你遇到過(guò)的奇怪寫(xiě)法
    2023-03-03
  • React進(jìn)階學(xué)習(xí)之組件的解耦之道

    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-08
  • ReactNative Image組件使用詳解

    ReactNative Image組件使用詳解

    本篇文章主要介紹了ReactNative Image組件使用詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-08-08
  • react如何獲取state的值并更新使用

    react如何獲取state的值并更新使用

    這篇文章主要介紹了react如何獲取state的值并更新使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • 使用useEffect模擬組件生命周期

    使用useEffect模擬組件生命周期

    這篇文章主要介紹了使用useEffect模擬組件生命周期,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-09-09
  • React?Native集成支付寶支付的實(shí)現(xiàn)方法

    React?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)化

    本篇文章主要介紹了淺談React + Webpack 構(gòu)建打包優(yōu)化,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-01-01
  • React函數(shù)式組件的性能優(yōu)化思路詳解

    React函數(shù)式組件的性能優(yōu)化思路詳解

    這篇文章主要介紹了React函數(shù)式組件的性能優(yōu)化思路詳解,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-04-04
  • 詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter

    詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter

    這篇文章主要介紹了詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-12-12
  • React?Native采用Hermes熱更新打包方案詳解

    React?Native采用Hermes熱更新打包方案詳解

    這篇文章主要介紹了React?Native采用Hermes熱更新打包實(shí)戰(zhàn),在傳統(tǒng)的熱更新方案中,我們實(shí)現(xiàn)熱更新需要借助code-push開(kāi)源方案,包括熱更新包的發(fā)布兩種方式詳解,感興趣的朋友一起看看吧
    2022-05-05

最新評(píng)論