九色国产,午夜在线视频,新黄色网址,九九色综合,天天做夜夜做久久做狠狠,天天躁夜夜躁狠狠躁2021a,久久不卡一区二区三区

打開APP
userphoto
未登錄

開通VIP,暢享免費電子書等14項超值服

開通VIP
利用 React/Redux/React
  1. 如何設(shè)計一個大型 web 項目?
  2. React + webpack 如何按需加載?
  3. React + React-Router 4 + webpack 如何按需加載?
  4. React + Redux + React-Router 4 + webpack 如何按需加載?

實錄提要:

  • bundle-loader 和 Webpack 內(nèi)置的 import() 有什么區(qū)別?
  • 按需加載能否支持通過請求后臺數(shù)據(jù),動態(tài)配置頁面的的應(yīng)用場景?
  • 參與過幾個 React 項目,被依賴包搞的暈暈的,不知道該怎么選擇?
  • 什么包應(yīng)該放到 devDependencies 里面?什么包放到 depedencies 里面?
  • 為什么是 react-router-redux 而不是傳統(tǒng)的 react-redux,其優(yōu)勢是什么?
  • 按需加載時,每個單獨的 bundle 都挺大的,為什么?
  • ECMAScript 每年出一個版本,對應(yīng)的 babel 也有一大堆,應(yīng)該如何選擇?
  • 單頁項目過大,怎么拆分不同模塊頁面到不同 js 來動態(tài)加載?

題外話

經(jīng)驗尚淺,尚不足以教導(dǎo),若理解有誤,望能指導(dǎo)三分,語言若有偏激,請理解我年輕氣盛。

之所以寫這篇文章,是因為我最近一陣子經(jīng)歷了一個部門的技術(shù)選型->項目實施這些技術(shù)->二次技術(shù)選型->技術(shù)版本升級的一個過程。開發(fā)業(yè)務(wù)應(yīng)用為主的我們,很少有時間去研究某項技術(shù)的源碼,不加班趕項目進(jìn)度就已經(jīng)很慶幸了,大部分時間都花在了如何靈活使用市面上的一些技術(shù)體系。在這篇文章中,不涉及源碼范圍,我也沒去研究過源碼。寫這篇文章的初衷是分享我的想法和代碼示例,同時也希望看這篇文章的你能夠給予寶貴的意見,讓我得以進(jìn)步。

web應(yīng)用讓人驚嘆是從Gmail開始的,流暢的桌面版體驗吸引了很多人,從此web項目開始蓬勃發(fā)展。隨后,web應(yīng)用也越來越復(fù)雜,為了能讓web應(yīng)用如同桌面版應(yīng)用一樣流暢,出現(xiàn)了SPA。這就是今天我想說的,react/redux等等一系列的產(chǎn)品的出現(xiàn)都是為了實現(xiàn)體驗度更佳的SPA。

兩年前,我開發(fā)web項目,都只是用javaweb,使用模板引擎,后端渲染出頁面。對于訪問量不是很大、單個頁面復(fù)雜度不是很高、項目的迭代周期不頻繁、二次開發(fā)的次數(shù)很少的系統(tǒng),這種模式無疑很適用、性能也沒什么大的影響,一個java程序員就可以做到全棧。而事實上,我手頭的web項目并非這么簡單,隨著周期的迭代,項目越來越臃腫,后端代碼和前端代碼摻雜在一起混亂不堪,當(dāng)時我自認(rèn)為自己技術(shù)不錯,代碼寫的自己都認(rèn)識,然而我離職之后發(fā)現(xiàn)一個很多人都知道的道理,一個技術(shù)真正好的程序員,寫出來的代碼是要能夠讓他人讀的懂,最起碼要讓接替你繼續(xù)這個項目的人讀得懂,技術(shù)不行那是例外,當(dāng)時我沒有做到。而后,進(jìn)入新的公司,讓我能夠有機(jī)會去對部門進(jìn)行技術(shù)選型,主要是前端部分。我果斷選擇了前后端分離模式,我不想以前不好的地方繼續(xù)發(fā)生在以后。人員安排最好是這樣,專職的人做專職的事,全棧人員要能夠補位。設(shè)計一個優(yōu)質(zhì)的web項目,最重要的是人,而不是技術(shù)!

設(shè)計web最好是前端設(shè)計成SPA、后端設(shè)計成微服務(wù),有很多企業(yè)使用react、vue、angular這些,結(jié)果是多頁應(yīng)用,增加復(fù)雜度,降低頁面切換的流暢度。我真不知道他們是怎么想的,首先多頁應(yīng)用是不可取的,如果他們是開發(fā)webapp,封裝成apk,那就更不可取了!web項目設(shè)計成SPA,很多人會想,隨著代碼量的增大,首次加載的文件就會增大,沒錯,這時就需要用到code splitting。這也就是我今天要講的實際項目中如何進(jìn)行按需加載。我見過有些開發(fā)人員將一個js文件拆分成多個js文件,而每個頁面都加載這些js文件,這顯然是不可取的,這樣子的不需要拆分。隨著現(xiàn)在網(wǎng)速的提升,首次加載文件稍微大點都是可以接受的。首次加載后緩存在瀏覽器處,下次加載的時候會更快。引入第三方UI組件,基礎(chǔ)組件要單一,不可以引入antd了再去引用bootstrap,還有用了react這些就不要在項目中出現(xiàn)jQuery,要純粹!

廢話就講到這里,下面介紹我將一個項目重構(gòu)三次的過程,這三個過程里有三種不同的按需加載方式。示例是我將實際項目刪減過后可運行的例子。code splitting和react/react-router是沒有直接關(guān)系的。

一、react(v.0.14.8) / react-router(v.1.0.3) / webpack(v.1.13.3)

兼容IE8+及現(xiàn)代瀏覽器,示例代碼地址->https://github.com/love-fay/fay-webpack-redux-code-splitting/tree/master/react。

先貼下依賴:

因為業(yè)務(wù)的需求,需要兼容到IE8,不得不被動地選擇低版本庫。處于對react的首次使用,并沒有加入redux相關(guān)技術(shù)。

在react-router 1.x版本中Route組件上擁有g(shù)etComponent、onEnter參數(shù)(4.x之后被移除),getComponent是異步的,所以我們可以在這個參數(shù)里進(jìn)行按需加載,getComponent這個函數(shù)有兩個參數(shù)nextState、callback,根據(jù)nextState.pathname可以獲取到路由地址,然后再利用webpack的require.ensure異步加載所屬組件的js文件,最后通過callback將該組件返回。示例代碼如下:

<Route path="app" getComponent={this.getUumsComponent} onEnter={this.requireAuth}/>
getUumsComponent = (nextState, callback) ={    let pathname = nextState.pathname;    switch (pathname) {        case 'app':        require.ensure([], (require) ={            callback(null, require('../app/components/App'));                }, 'App');            break;        default :            historyConfig.pushState({nextPathname: pathname}, '/404');    }};

打包過后主要文件的對比:

code splitting

not code splitting

這種方式的按需加載就這些,很簡單~

二、react(v.15.6.1) / react-router(v.4.2.2)b / webpack(v.3.5.6)

兼容IE9+及現(xiàn)代瀏覽器,示例代碼地址->https://github.com/love-fay/fay-webpack-redux-code-splitting/tree/master/react-rr4。

先貼下依賴:

這次決定拋棄IE8,甚至都不想兼容IE。很多人口口聲聲說用戶體驗、用戶需求,卻一味地去支持IE8,甚至還有支持IE6的!其實用戶體驗和需求不是用戶單方面的要求,還有就是開發(fā)方需要去改變用戶習(xí)慣、引導(dǎo)用戶對未來的需求,在這基礎(chǔ)上不斷地提高用戶體驗。(你不告知你的用戶有個瀏覽器叫做谷歌瀏覽器,他這輩子就會覺得IE就是瀏覽器,瀏覽器就是IE。)

這次主要是將react/react-router/webpack進(jìn)行了升級,并升級到最新(當(dāng)時的最新)。

按需加載其實跟react-router沒多大關(guān)系,只不過需要借助它更好的完成按需加載這項任務(wù)。react-router升級到4后,便沒有了getComponent這個參數(shù),所以我們得換種方式,react-router4官方示例也提供了code splitting的方法,利用webpack結(jié)合bundle-loader,它是在require.ensure基礎(chǔ)上封裝的,更友好的實現(xiàn)異步加載過程。

bundle-loader可以在webpack文件中進(jìn)行配置,這里我就不介紹了,webpack官方文檔都有寫。我這里是寫在代碼里的。我簡單說下,基本跟react-router4官方文檔說的差不多。

首先先寫一個bundle.js這個組件,代碼如下:

import React, { Component } from 'react';import PropTypes from 'prop-types';class Bundle extends Component {    static propTypes = {        load: PropTypes.any,        children: PropTypes.any,    };    state = {        mod: null,    };    componentWillMount () {        this.load(this.props);    }    componentWillReceiveProps (nextProps) {        if (nextProps.load !== this.props.load) {            this.load(nextProps);        }    }    load (props) {        this.setState({            mod: null,        });        props.load((mod) ={            this.setState({                mod: mod['default'] ? mod['default'] : mod,            });        });    }    render () {        return this.state.mod ? this.props.children(this.state.mod) : <div></div>;    }}export default Bundle;

然后在用到需要按需加載的組件的組件中,引入的時候,在文件路徑前面使用bundle-loader?lazy&name=[App]!,如下:

import loadApp from 'bundle-loader?lazy&name=[App]!../../app/components/App';

然后比如我這里使用 <Route path="/app" component={App}/> 加載這個App組件,我們需要用到剛才自己寫的bundle組件:

import Bundle from '../bundle/components/Bundle';const App = (props) =(    <Bundle load={loadApp}>        {(App) ={            return <App {...props}/>;        }}    </Bundle>);

打包過后主要文件的對比:

code splitting

not code splitting

到這里,這第二種方式介紹完了,很簡單~

二、react(v.16.1.1) / redux(v.3.7.2) / react-router(v.4.2.2) / webpack(v.3.8.1)

兼容IE9+及現(xiàn)代瀏覽器,示例代碼地址->https://github.com/love-fay/fay-webpack-redux-code-splitting/tree/master/react-rr4-redux。

先貼下依賴:

這次又一次對引用的技術(shù)進(jìn)行了更新,同時加入了redux,項目復(fù)雜度的提高,組件之間的交流變得復(fù)雜,此時就需要用到redux。有些開發(fā)人員會覺得好煩,不斷地升級,不斷地改造,很費時費力,什么時候才能穩(wěn)定,其實不然,項目的穩(wěn)定不代表技術(shù)的不變,穩(wěn)定是相對的。如果想要一勞永逸的話,就不要讓公司給你漲工資了,公司也想一勞永逸~以后人工智能一旦鋪開到企業(yè)級開發(fā)中,將會導(dǎo)致大量在安逸中度過的程序員失業(yè)!學(xué)習(xí)是無止境的,學(xué)習(xí)也是人一輩子免費的技能,曾經(jīng)后端Java一家獨大的時候,spring3穩(wěn)定的時候,很多后端程序員就開始陷入了一勞永逸的幻覺當(dāng)中,導(dǎo)致他們中的很多人一度抱怨前端是在瞎折騰~這就好比有自行車為什么要造汽車的理論是一樣的~我是以Java程序員入行的,很清楚Java寫后端的時候,輪子很多,很多程序員就是使用CV大法,甚至很多項目經(jīng)理啊什么的就說程序員是搬運工,代碼不就是增刪改查么~

使用了redux后,全局只有一個Store,而這個Store在頁面打開的時候就已經(jīng)聲明了,于是讓我很糾結(jié)如何按需加載。后來我了解到redux這個東西的存在,內(nèi)部運用了react中的context,同時這個context算是隱藏著的秘密。利用它我可以改變?nèi)值腟tore。我這里使用了react-redux,在頂級組件處加入。

import {Provider} from 'react-redux';<Provider store={store}>    ......</Provider>

然后在需要引入store信息的子組件處利用它提供的connect方法將store派發(fā)下去,這里派發(fā)是根據(jù)上下文context。項目中少不了用到路由,這時候,我使用了react-router-redux(一定要5.x版本npm i react-router-redux@next),在總的reducer中加入routerReducer,然后在寫路由組件的部分的頂級處使用。

import createBrowserHistory from 'history/createBrowserHistory';import { ConnectedRouter} from 'react-router-redux';const history = createBrowserHistory();<ConnectedRouter history={history}>    ......</ConnectedRouter>

讓我們再回到上一個代碼片,其中的store來源如下:

import configureStore from '../Store';let store = configureStore();

Store.js

import {createStore, applyMiddleware, compose} from 'redux';import { createLogger } from 'redux-logger';const logger = createLogger();import { routerMiddleware } from 'react-router-redux';import createHistory from 'history/createBrowserHistory';import createSagaMiddleware from 'redux-saga';const history = createHistory();const rMiddleware = routerMiddleware(history);const win = window;export const sagaMiddleware = createSagaMiddleware();const middlewares = [rMiddleware, sagaMiddleware];if (process.env.NODE_ENV !== 'production') {    middlewares.push(require('redux-immutable-state-invariant').default());}const storeEnhancers = compose(    applyMiddleware(...middlewares, logger),    (win && win.devToolsExtension) ? win.devToolsExtension() : (f) =f,);import createReducer from './reducers';export function injectAsyncStore(store, asyncReducers, sagas) {    asyncReducers && injectAsyncReducers(store, asyncReducers);    sagas && injectAsyncSagas(store, sagas);}function injectAsyncReducers(store, asyncReducers) {    let flag = false;    for (let key in asyncReducers) {        if(Object.prototype.hasOwnProperty.call(asyncReducers, key)) {            if (!store.asyncReducers[key]) {                store.asyncReducers[key] = asyncReducers[key];                flag = true;            }        }    }    flag && store.replaceReducer(createReducer(store.asyncReducers));}function injectAsyncSagas(store, sagas) {    for (let key in sagas) {        if(Object.prototype.hasOwnProperty.call(sagas, key)) {            if (!store.asyncSagas[key]) {                store.asyncSagas[key] = sagas[key];                store.sagaMiddleware.run(sagas[key]);            }        }    }}export default function configureStore() {    let store = createStore(createReducer(), {}, storeEnhancers);    store.asyncReducers = {};    store.asyncSagas = {};    store.sagaMiddleware = sagaMiddleware;    return store;}

reducers.js

import { combineReducers } from 'redux';import { routerReducer } from 'react-router-redux';export default function createReducer(asyncReducers) {    const reducers = {        ...asyncReducers,        router: routerReducer    };    return combineReducers(reducers);}

我沒有進(jìn)行刪減,主要是動態(tài)改變store中兩個東西,一個是reducer還有一個就是saga。異步請求這塊我用的是redux-saga,雖然官方文檔上露臉的是redux-thunk和redux-promise,但是后起之秀redux-saga做到低耦合,在項目中作為獨立的一層出現(xiàn),不與action creator和reducer耦合。還有就是它強(qiáng)大的異步流程控制。

再來看看路由部分是怎么寫的:

<Provider store={store}>    <ConnectedRouter history={history}>        <Switch>            <Route path='/app' component={App}/>        </Switch>    </ConnectedRouter></Provider>

這里的App組件便是我們要按需加載的組件。我是按照模塊來組織我的代碼的,先來看下App模塊的代碼排版:

這張圖中sagas.js是用來處理異步請求的,bundle.js和lazy.js以及公用的bundle.js是用來完成code splitting的。

lazy.js【需要懶加載的文件】

import appSagas from './sagas';import appReducer from './reducer';import view from './views/app';const reducer = {    appReducer: appReducer};const sagas = {    appSagas: appSagas};export {sagas, reducer, view};

bundle.js【code splitting】

import React from 'react';import Bundle from '../../bundle/views/bundle';import load from 'bundle-loader?lazy&name=[App]!./bundle';import {injectAsyncStore} from '../../Store';export default (props) ={    return (        <Bundle load={(store, cb) ={            load((target) ={                const {reducer, view, sagas} = target;                injectAsyncStore(store, reducer, sagas);                cb(view);            })        }}>            {(View) ={                return <View {...props}/>            }}        </Bundle>    );};

公用的bundle.js【對其進(jìn)行了改造,加入了store】

import React, { Component } from 'react';import PropTypes from 'prop-types';class Bundle extends Component {    static propTypes = {        load: PropTypes.any,        children: PropTypes.any,    };    static contextTypes = {        store: PropTypes.object    };    state = {        mod: null,    };    componentWillMount () {        this._isMounted = true;        this.load(this.props);    }    componentWillUnmount() {        this._isMounted = false;    }    componentWillReceiveProps (nextProps) {        if (nextProps.load !== this.props.load) {            this.load(nextProps);        }    }    load (props) {        this.setState({            mod: null,        });        props.load(this.context.store, (mod) ={            if (this._isMounted) {                this.setState({                    mod: mod['default'] ? mod['default'] : mod,                });            }        });    }    render () {        return this.state.mod ? this.props.children(this.state.mod) : <div>組件加載中...</div>;    }}export default Bundle;

index.js【對外暴露的組件】

import view from './bundle';export {view};

code splitting就完成了~當(dāng)然我又進(jìn)行了更改,下文有講。

這里要說下公用的bundle.js中的this.context.store,這里一定要定義contextTypes,不然獲取不到this.context,當(dāng)然官方?jīng)]有提供這個api,也不推薦使用,但是按需加載就得需要它,并且我們要謹(jǐn)慎使用它即可,因為this.context一旦改變,它關(guān)聯(lián)的上下文就會重新render,所以加載某個頁面的時候,把它所要使用到的reducer和sagas也都關(guān)聯(lián)進(jìn)去,這樣加載這個頁面其他組件的時候就已經(jīng)存在相關(guān)的reducer和sagas,不需要再改變上下文的store了。還有組件設(shè)計很重要,如果不合理會導(dǎo)致頁面不可控。lazy.js中的reducer和sagas是個對象,比如app這個組件中如果嵌套了其他組件,而這些其他組件中需要引入reducer和sagas,這時可以將這些reducer和sagas結(jié)合到app模塊的lazy.js中。不加入也可以,這時需要靈活運用shouldComponentUpdate這個生命周期來控制頁面。

從上面的代碼中,可以發(fā)現(xiàn)每個模塊的bundle.js中存在類似的代碼,這樣我們可以給其剝離出來,這不是必須的,因為剝離出來后,我們需要約定好每個模塊lazy.js中必須是export {sagas, reducer, view},當(dāng)然也可以約定其他,一致就行。這樣代碼進(jìn)過改造后,每個模塊的bundle.js代碼就可以分離到公共的bundle.js和模塊中的index.js中,代碼如下。

bundle.js中改動的代碼片:

load (props) {    this.setState({        mod: null,    });    props.load((mod) ={        const {reducer, view, sagas} = mod;        injectAsyncStore(this.context.store, reducer, sagas);        if (this._isMounted) {            this.setState({                mod: view['default'] ? view['default'] : view,            });        }    });}

index.js【以app模塊為例】

import React from 'react';import Bundle from '../../bundle/views/bundle';import load from 'bundle-loader?lazy&name=[App]!./lazy';const view = (props) ={    return (        <Bundle load={load}>            {(View) ={                return <View {...props}/>            }}        </Bundle>    );};export {view};

打包過后主要文件的對比:

code splitting

not code splitting

關(guān)于組件設(shè)計,使用reactjs的時候組件設(shè)計一定要足夠的扁平化,也就是平級,這樣不僅提高了計算的效率,同時也會很少出現(xiàn)父組件中嵌套子組件,而父組件更新的時候,子組件也跟著更新,實際上子組件并不想更新。當(dāng)然遇到逼不得已嵌套的情況的時候,可以使用shouldComponentUpdate這個組件存在時期的生命周期來控制子組件是否render??上驳氖莚eact16版本中B組件不嵌套在A組件中,渲染后出現(xiàn)在A組件里,也可以掛載到任何一個組件里,這就是portals。

看到這里,你會發(fā)現(xiàn)其實code splitting跟react和react-router沒多大關(guān)系,直接的聯(lián)系是redux和webpack,所以這種方式同時也適用于其他使用redux和webpack這種類似的技術(shù)體系。

react技術(shù)棧是目前前端最美的技術(shù)棧。

本站僅提供存儲服務(wù),所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊舉報。
打開APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
React,一次學(xué)習(xí),到處編碼
【第598期】ReactJS 服務(wù)端同構(gòu)實踐
我們?yōu)槭裁葱枰猂eact?
【第651期】React.js 初學(xué)者應(yīng)該知道的 9 件事
AngularJS遷移之戰(zhàn)
深入實戰(zhàn):構(gòu)建現(xiàn)代化的Web前端應(yīng)用
更多類似文章 >>
生活服務(wù)
熱點新聞
分享 收藏 導(dǎo)長圖 關(guān)注 下載文章
綁定賬號成功
后續(xù)可登錄賬號暢享VIP特權(quán)!
如果VIP功能使用有故障,
可點擊這里聯(lián)系客服!

聯(lián)系客服