我是如何使用React+Redux構建大型應用的

背景

咱們團隊有個項目因爲開發時間較長,且是先後端雜糅的開發方式,維護成本很高,在線上暴露的問題不少。並且由於對接了公司一百多條產品線,天天都會接到大量的客訴和產品線反饋的問題。2017年11月份以產品爲主導,從產品層面對流程進行從新設計,對該項目進行了先後端的重構。做爲前端的負責人我用這篇文章分享下,整個過程從技術選型,開發,上線的一些經驗。javascript

技術選型的思考

首先咱們先看下下面咱們項目中的幾個頁面,來總結下一些他們的特色。前端

咱們的頁面主要是須要用戶填寫的表單居多,在頁面加載的時候不須要去請求獲取和渲染大量的數據。並且一個頁面須要顯示的狀態較多(好比上面的3張圖,在項目中是一個組件)。還有一個最主要的業務需求,百度公司內部產品線較多,不一樣的業務都有其獨特的帳號標籤,這些帳號除了會走一些通用流程還要走一些對應產品線特點的流程。java

結合這些業務特點和以前有Nodejs和React的開發經驗,我總體的一個技術選型是FIS3+Nodejs+React+Redux+React-Router。那麼這些技術選型能帶來什麼呢?react

  1. 前端能夠在瀏覽器端控制頁面跳轉的路由,增長了前端開發的靈活性;
  2. 頁面能夠根據業務需求在服務選擇模板引擎渲染或者是同構渲染;
  3. 前端對錯誤碼文案和頁面文案作統一的管理,並且經過Nodejs來實現線下「熱更新」他們,線上實時生效;
  4. 有了Redux以後,作跨組件(多頁面)的數據共享更加方便。減小無心義的網絡請求。提升項目運行的穩定性和可用性。

這裏簡單的聊下工程化工具的選擇。目前在業內最火的工程化工具就是Webpack了吧。除了看過文檔以外,並無太多的實際應用經驗。我一直認爲使用工具就是來幫助開發者解決一些開發過程當中遇到的一些須要人爲頻繁去操做的無異議的工做。拋開Webpack咱們依舊能夠手動去編譯代碼,手動部署,手動刷新頁面來開發,使用工具只是讓這一系列的流程可以連貫起來,下降開發成本。後端

在個人全部跟公司有關的項目中選擇的都是FIS3,我也認爲他足夠的好用,能知足我各色各樣的工程化需求。我並非排斥Webpack。我只是尚未找到一個理由,讓我選擇放棄如今使用的FIS3去使用Webpack。瀏覽器

新老框架機制的區別

這裏簡單介紹下,決定了技術選型以後,對於渲染頁面渲染機制的一些區別。安全

以前舊項目使用PHP+Smarty的渲染模式,將頁面在服務端渲染完成以後再統一吐給前端瀏覽器。而使用新的技術架構以後,咱們渲染頁面的方式更加的靈活。能夠選擇在服務端渲染,能夠徹底交給瀏覽器渲染,能夠同構渲染。由於咱們的頁面在首屏的時候不須要加載大量的數據,因此我仍是讓大部分頁面在瀏覽器端進行渲染。服務器

還有一種區別就是以前全部來自用戶的請求都會落到PHP的服務器上去。而新框架的請求都會落到前端的Nodejs服務器上去。因此前端工程師不只僅是寫好頁面和作好兼容性。對前端工程師的技術能力也會帶來考驗。網絡

React帶給前端的便利

前端控制路由渲染頁面

前面談的技術選型已經提到了使用React-Router來作頁面路由控制。並且React-Router提供了異步加載組件的功能,這爲咱們上線優化頁面的異步加載提供了技術基礎。前端工程師

<Route path="/v4/appeal/fillname" component={FillName} />
{* 這裏對某些組件作異步加載 *}
<Route
    path="/v4/appeal/selectuser"
    getComponent={selectUser()}
/>
        
function selectUser() {
    return (location, cb) => {
            require(['../accountselect/container/AccountSelect'], function (component) {
                cb(null, component);
            });
        };
    }

經過React-Router來作路由控制除了前端代碼以外,服務端也許呀作些配置。否則咱們的頁面在回退的時候就會出現問題(頁面找不到路由)。其實就是在咱們一般說的action成面作下路由控制,由於我使用的是Nodejs,因此個人配置下面這樣子的。

router.get('/fillname', router.action('index'));
router.get('/selectuser', router.action('index'));

事件

在前端事件由於開源協議的問題曾經短暫使用過Preact。React和Preact最大的區別就是對於一些事件的封裝。這些形成了Preact相對於React體積小不少。
作移動端開發,前端常常會面臨的一個問題就是click事件 300ms 延時的問題。在React中提供的onClick事件一樣也會出現這樣的問題。若是若是咱們想要在點擊一個按鈕以後,在其它地方當即出現反饋,最好就是使用onTouchEnd事件,或者就是使用開源的Npm包react-fastclick能很好的解決click事件 300ms延時的問題。

使用的方法就是在咱們代碼的入口地方,聲明如下語句,他默認會改變react的onClick事件的行爲

import initReactFastclick from 'react-fastclick';

initReactFastclick();

組件的設計

在使用React的時候可能都會面臨的問題,個人組件應該是無狀態的仍是有狀態的。個人組件狀態怎麼共享。何時我應該使用Redux來管理組件的狀態。剛開始接觸react都會有這樣的疑問吧。

一種比較極端的作法就是,無論狀態需不須要共享,組件的全部狀態都試用Redux來管理。這樣的作法就是咱們須要寫大量的Action。若是是一兩個頁面還好,若是是十幾個頁面,真的寫action是能把人寫崩潰的。

那麼最佳實踐是什麼呢?看下圖

當咱們要寫一個組件的時候,首先想下這個組件是否是須要與其它組件共享它自己的狀態。若是須要咱們應該把它當作有狀態的組件來設計,並且共享的狀態使用Redux來管理。若是簡單的就是無狀態組件或者是這個組件自己的狀態改變不會影響其它的組件,就能夠將組件設計爲無狀態組件(雖然叫無狀態組件,其實組件自己的狀態也是可使用this.state來管理的)。

組件的複用關係

React的一大熱點就是組件化的開發思想。小到頁面上的一個按鈕都是能夠設計成一個組件。既然是組件咱們首先就應該考慮這個組件怎麼被其它組件複用。

舉個簡單的例子,在整個項目中都會用到的彈窗組件:

class AlertForm extends Component {
    constructor(props) {
        super(props);

        this.state = {
            showlayout: false,  // false 以tip的方式提示錯誤, true以彈層的方式提示錯誤
            btnlist: false,
            formbtn: false
        };
    }

    componentWillReceiveProps(nextProps) {
        
    }

    handleHideLayout = () => {
    }

    handleMobile = () => {
    }

    handleChangeCheck = () => {
        history.go(-1);
    }

    render() {

        return (
            <div className="component-alertform" style={this.state.showlayout ? {display: 'block'} : {display: 'none'}}>
                
            </div>
        );
    }
}

export default AlertForm;

咱們將這種可能在其餘頁面都用的組件單獨抽象成出來,在須要用的地方import

import AlertForm from '../../components/AlertForm';

<AlertForm
    errno={errno}
    stateObj={fillAppealName}
    actions={actions}
/>

開發環境和生產環境打包優化

完成項目以後確定要作的一項工做就是上下前的優化,上線前我作的工做主要以下:

前面已經談到錯對於大多數用戶來講都只是會走一些普通流程。有些具備產品線特點的用戶會走一些特殊流程。因此在上線前確定要拆包,和作組件的異步加載。具體的前面已經提到過了。在打包的時候對這些頁面的js須要使用打包工具作單獨的處理。

其實除了這些須要異步加載的頁面以外還會存在一些其餘本身編寫的lib庫(本身編寫的小函數)。還有好比全國省市地區對應關係,電話區號對應關係。由於這些函數或者是地區關係映射圖在上線之後基本上都是不會再變化的,因此與業務的js分開打包。

咱們的打包的配置文件以下:

運維

前面已經談到使用Nodejs作中間層,作路由控制和服務端渲染。下面的這張圖是我寫這篇文章的時候截取的額以上服務實時狀態圖。能夠發現,整個應用對於內存、磁盤IO利用率仍是很正常的,對於CPU的利用率有點兒高,這也是後續須要優化的地方。

這裏想要說的是,若是使用了Nodejs,使用了服務端渲染,對於前端工程師的我的素質要求會比較高,由於須要處理不少服務端的問題。前面也分享過一篇處理安全工單的問題,不只僅要面對服務端的問題,還有面對來自互聯網安全的問題。

其它能力補充

使用Nodejs除了來作服務端渲染。我還在使用Nodejs作了一些其它的工做。

好比我在服務端使用Nodejs管理了這樣一個JSON文件。PHP端不在維護錯誤碼和錯誤碼顯示的文案。全部前端須要顯示文案放在Nodejs端作統一的管理。並且,我線下也能夠同經過系統對這些錯誤文案進行動態的更新。提升系統的可用性。

相關文章
相關標籤/搜索