成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

精益 React 學(xué)習(xí)指南 (Lean React)- 4.2 react patterns

Berwin / 1107人閱讀

摘要:另外一點是組件應(yīng)該盡量保證獨立性,避免和外部的耦合,使用全局事件造成了和外部事件的耦合。明確的職責(zé)分配也增加了應(yīng)用的確定性明確只有組件能夠知道狀態(tài)數(shù)據(jù),且是對應(yīng)部分的數(shù)據(jù)。

書籍完整目錄

4.2 react patterns

修改 Props

Immutable data representation

確定性

在 getInitialState 中使用 props

私有狀態(tài)和全局事件

render 包含 side effects

jQuery 修改 DOM

使用無狀態(tài)組件

內(nèi)存管理

componentWillUnmount 取消訂閱事件

判斷 isMounted

上層設(shè)計

使用 container component

使用 Composition 替代 mixins

Composability - Presenter Pattern

Composability - Decorator Pattern

Context 數(shù)據(jù)傳遞

4.2.1 關(guān)于

React 的框架設(shè)計是趨于函數(shù)式的,其中最主要的兩點也是為什么會選擇 React 的兩點:

單向性:數(shù)據(jù)的流動是單向的

確定性:React(storeData) = view 相同數(shù)據(jù)總是渲染出相同的 view

這兩點即是特性也是設(shè)計 React 應(yīng)用的基本原則,圍繞這兩個原則社區(qū)里邊出現(xiàn)了一些 React 設(shè)計模式,即有好的設(shè)計模式也有應(yīng)該要避免的反模式,理解這些設(shè)計模式能夠幫助我們寫出更優(yōu)質(zhì)的 React 應(yīng)用,本節(jié)將圍繞 單向性、確定性、內(nèi)存管理、上層設(shè)計 來討論這些設(shè)計模式。

anti 表示反模式,good 表示好模式

4.2.2 單向性

數(shù)據(jù)的流動是單向的

修改 Props (anti)

描述: 組件任何地方修改 props 的值

解釋:

React 的數(shù)據(jù)流動是單向性的,流動的方式是通過 props 傳遞到組件中,而在 Javascript 中對象是通過引用傳遞的,修改 props 等于直接修改了 store 中的數(shù)據(jù),導(dǎo)致破壞數(shù)據(jù)的單向流動特性

使用不可變數(shù)據(jù) (good)

描述: store data 使用不可變數(shù)據(jù)

解釋: Javascript 對象的特性是可以任意修改,而這個特性很容易破壞數(shù)據(jù)的單向性,因為人工無法永遠(yuǎn)確保數(shù)據(jù)沒有被修改過,唯一的做法是使用不可變數(shù)據(jù),用代碼邏輯確保數(shù)據(jù)不能被任意修改,后面會有一個完整的小節(jié)介紹不可變數(shù)據(jù)在 React 中的應(yīng)用

4.2.3 確定性

React(storeData) = view 相同數(shù)據(jù)總是渲染出相同的 view

在 getInitialState 中使用 props (anti)

描述: getInitialState 通過 props 來生成 state 數(shù)據(jù)

解釋:

官方文檔 https://facebook.github.io/react/tips/props-in-getInitialState-as-anti-pattern.html

在 getInitialState 中通過 props 來計算 state 破壞了確定性原則,“source of truth” 應(yīng)該只是來自于一個地方,通過計算 state 過后增加了 truth source。這種做法的另外一個壞處是在組件更新的時候,還需要計算重新計算這部分 state。

舉例:

var MessageBox = React.createClass({
  getInitialState: function() {
    return {nameWithQualifier: "Mr. " + this.props.name};
  },

  render: function() {
    return 
{this.state.nameWithQualifier}
; } }); ReactDOM.render(, mountNode);

優(yōu)化方式:

var MessageBox = React.createClass({
  render: function() {
    return 
{"Mr. " + this.props.name}
; } }); ReactDOM.render(, mountNode);

需要注意的是以下這種做法并不會影響確定性

var Counter = React.createClass({
  getInitialState: function() {
    // naming it initialX clearly indicates that the only purpose
    // of the passed down prop is to initialize something internally
    return {count: this.props.initialCount};
  },

  handleClick: function() {
    this.setState({count: this.state.count + 1});
  },

  render: function() {
    return 
{this.state.count}
; } }); ReactDOM.render(, mountNode);
私有狀態(tài)和全局事件 (anti)

描述: 在組件中定義私有的狀態(tài)或者使用全局事件

介紹: 組件中定義了私有狀態(tài)和全局事件過后,組件的渲染可能會出現(xiàn)不一致,因為全局事件和私有狀態(tài)都可以控制組件的狀態(tài),這樣外部使用組件無法保證組件的渲染結(jié)果,影響了組件的確定性。另外一點是組件應(yīng)該盡量保證獨立性,避免和外部的耦合,使用全局事件造成了和外部事件的耦合。

render 函數(shù)包含 side effects (anti)

side effect 解釋: https://en.wikipedia.org/wiki/Side_effect_(computer_science)

描述: render 函數(shù)包含一些 side effects 的代碼邏輯,這些邏輯包括如

修改 state 數(shù)據(jù)

修改 props 數(shù)據(jù)

修改全局變量

調(diào)用其他導(dǎo)致 side effect 的函數(shù)

解釋: render 函數(shù)如果包含了 side effect ,渲染的結(jié)果不再可信,所以確保 render 函數(shù)為純函數(shù)

jQuery 修改 DOM (anti)

描述: 使用外部 DOM 框架修改或刪除了 DOM 節(jié)點、屬性、樣式
解釋: React 中 DOM 的結(jié)構(gòu)和屬性都是由渲染函數(shù)確定的,如果使用了 Jquery 修改 DOM,那么可能造成沖突,視圖的修改源頭增加,直接影響組件的確定性

使用無狀態(tài)組件 (good)

描述: 優(yōu)先使用無狀態(tài)組件
解釋: 無狀態(tài)組件更符合函數(shù)式的特性,如果組件不需要額外的控制,只是渲染結(jié)構(gòu),那么應(yīng)該優(yōu)先選擇無狀態(tài)組件

4.2.4 內(nèi)存管理 componentWillUnmount 取消訂閱事件 (good)

描述: 如果組件需要注冊訂閱事件,可以在 componentDidMount 中注冊,且必須在 ComponentWillUnmount 中取消訂閱
解釋: 在組件 unmount 后如果沒有取消訂閱事件,訂閱事件可能仍然擁有組件實例的引用,這樣第一是組件內(nèi)存無法釋放,第二是引起不必要的錯誤

判斷 isMounted (anti)

描述: 在組件中使用 isMounted 方法判斷組件是否未被注銷
解釋:

React 中在一個組件 ummount 過后使用 setState 會出現(xiàn)warning提示(通常出現(xiàn)在一些事件注冊回調(diào)函數(shù)中) ,避免 warning 的解決辦法是:

if(this.isMounted()) { // This is bad.
  this.setState({...});
}

但這是個掩耳盜鈴的做法,因為如果出現(xiàn)了錯誤提示就表示在組件 unmount 的時候還有組件的引用,這個時候應(yīng)該是已經(jīng)導(dǎo)致了內(nèi)存溢出。所以解決錯誤的正確方法是在 componentWillUnmount 函數(shù)中取消監(jiān)聽:

class MyComponent extends React.Component {
  componentDidMount() {
    mydatastore.subscribe(this);
  }
  render() {
    ...
  }
  componentWillUnmount() {
    mydatastore.unsubscribe(this);
  }
}
4.2.5 上層設(shè)計 使用 container component (good)

描述: 將 React 組件分為兩類 container 、normal ,container 組件負(fù)責(zé)獲取狀態(tài)數(shù)據(jù),然后傳遞給與之對應(yīng)的 normal component,對應(yīng)表示兩個組件的名稱對應(yīng),舉例:

TodoListContainer => TodoList
FooterContainer => Footer

解釋: 參看 redux 設(shè)計中的 container 組件,container 組件是 smart 組件,normal 組件是 dummy 組件,這樣的責(zé)任分離讓 normal 組件更加獨立,不需要知道狀態(tài)數(shù)據(jù)。明確的職責(zé)分配也增加了應(yīng)用的確定性(明確只有 container 組件能夠知道狀態(tài)數(shù)據(jù),且是對應(yīng)部分的數(shù)據(jù))。

使用 Composition 替代 mixins (good)

描述: 使用組件的組合的方式(高階組件)替代 mixins 實現(xiàn)為組件增加附加功能
解釋:

mixins 的設(shè)計主要目的是給組件提供插件機(jī)制,大多數(shù)情況使用 mixin 是為了給組件增加額外的狀態(tài)。但是使用 mixins 會帶來一些額外的壞處:

mixins 通常需要依賴組件定義特定的方法,如 getSomeMixinState ,而這個是隱式的約束

多個 mixins 可能會導(dǎo)致沖突

mixins 通常增加了額外的狀態(tài)數(shù)據(jù),而 react 的設(shè)計應(yīng)該是要避免過多的內(nèi)部狀態(tài)

mixins 可能會影響 shouldComponentUpdate 的邏輯, mixins 做了很多數(shù)據(jù)合并的邏輯

另外一點是在新版本的 React 中,mixins 將會是廢棄的 feature,在 es6 class 定義組件也不會支持 mixins。

舉個例子,一個訂閱 fluxstore 的 mixin 為:

function StoreMixin(store) {
  var Mixin = {
    getInitialState() {
      return this.getStateFromStore(this.props);
    },
    componentDidMount() {
      store.addChangeListener(this.handleStoreChanged)
      this.setState(this.getStateFromStore(this.props));
    },
    componentWillUnmount() {
      store.removeChangeListener(this.handleStoreChanged)
    },
    handleStoreChanged() {
      if (this.isMounted()) {
        this.setState(this.getStateFromStore(this.props));
      }
    }
  };
  return Mixin;
}

使用

const TodolistContainer = React.createClass({
  mixins: [StoreMixin(AppStore)],
  getStateFromStore(props) {
    return {
      todos: AppStore.get("todos");
    }
  }
})

轉(zhuǎn)換為組件的組合方式為:

function connectToStores(Component, store, getStateFromStore) {
  const StoreConnection = React.createClass({
    getInitialState() {
      return getStateFromStore(this.props);
    },
    componentDidMount() {
        store.addChangeListener(this.handleStoreChanged)
    },
    componentWillUnmount() {
        store.removeChangeListener(this.handleStoreChanged)
    },
    handleStoreChanged() {
      if (this.isMounted()) {
        this.setState(getStateFromStore(this.props));
      }
    },
    render() {
      return ;
    }
  });
  return StoreConnection;
};

使用方式:

class Todolist extends React.Component {
    render() {
        // ....
    }
}
TodolistContainer = connectToStore(Todolist, AppStore, props => {
    todos: AppStore.get("todos")
})
Presenter Pattern

描述: 利用 children 可以作為函數(shù)的特性,將數(shù)據(jù)獲取和數(shù)據(jù)表現(xiàn)分離成為兩個不同的組件

如下例子:

class DataGetter extends React.Component {
  render() {
    const { children } = this.props
    const data = [ 1,2,3,4,5 ]
    return children(data)
  }
}

class DataPresenter extends React.Component {
  render() {
    return (
      
        {data =>
          
    {data.map((datum) => (
  • {datum}
  • ))}
}
) } } const App = React.createClass({ render() { return ( ) } })

解釋: 將數(shù)據(jù)獲取和數(shù)據(jù)展現(xiàn)分離,同時利用組件的 children 可以作為函數(shù)的特性,讓數(shù)據(jù)獲取和數(shù)據(jù)展現(xiàn)都可以作為組件使用

Decorator Pattern

描述: 父組件通過 cloneElement 方法給子組件添加方法和屬性

cloneElement 方法:

ReactElement cloneElement(
  ReactElement element,
  [object props],
  [children ...]
)

如下例子:

const CleverParent = React.createClass({
  render() {
    const children = React.Children.map(this.props.children, (child) => {
      return React.cloneElement(child, {
        // 新增 onClick 屬性
        onClick: () => alert(JSON.stringify(child.props, 0, 2))
      })
    })
    return 
{children}
} }) const SimpleChild = React.createClass({ render() { return (
{this.props.children}
) } }) const App = React.createClass({ render() { return ( 1 2 ) } })

解釋: 通過這種設(shè)計模式,可以應(yīng)用到一些自定義的組件設(shè)計,提供更簡潔的 API 給第三方使用,如 facebook 的 FixedDataTable 也是應(yīng)用了這種設(shè)計模式

Context 數(shù)據(jù)傳遞

描述: 通過 Context 可以讓所有組件共享相同的上下文,避免數(shù)據(jù)的逐級傳遞, Context 是大多數(shù) flux 庫共享 store 的基本方法。

使用方法:

/**
 * 初始化定義 Context 的組件
 */
class Chan extends React.Component {
  getChildContext() {
    return {
      environment: "grandma"s house"
    }
  }
}

// 設(shè)置 context 類型
Chan.childContextTypes = {
  environment: React.PropTypes.string
};

/**
 * 子組件獲取 context 
 */
class ChildChan extends React.Component {
  render() {
    const ev = this.context.environment;
  }
}
/**
 * 需要設(shè)置 contextTypes 才能獲取
 */
ChildChan.contextTypes = {
  environment: React.PropTypes.string
};

解釋: 通常情況下 Context 是為基礎(chǔ)組件提供的功能,一般情況應(yīng)該避免使用,否則濫用 Context 會影響應(yīng)用的確定性。

參考鏈接

https://medium.com/@dan_abramov/mixins-are-dead-long-live-higher-order-components-94a0d2f9e750#.hvbsii4zd

http://www.zhubert.com/blog/2016/02/05/react-composability-patterns/

https://medium.com/@learnreact/context-f932a9abab0e#.wn00ktlde

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/79819.html

相關(guān)文章

  • 精益 React 學(xué)習(xí)指南Lean React)- 4.1 react 代碼規(guī)范

    書籍完整目錄 4.1 react 代碼規(guī)范 showImg(https://segmentfault.com/img/bVyE9m); 關(guān)于 基礎(chǔ)規(guī)范 組件結(jié)構(gòu) 命名規(guī)范 jsx 書寫規(guī)范 eslint-plugin-react 關(guān)于 在代碼的設(shè)計上,每個團(tuán)隊可能都有一定的代碼規(guī)范和模式,好的代碼規(guī)范能夠提高代碼的可讀性便于協(xié)作溝通,好的模式能夠上層設(shè)計上避免不必要的 bug 出現(xiàn)。本節(jié)會參考...

    zhangqh 評論0 收藏0
  • 精益 React 學(xué)習(xí)指南Lean React)- 1.2 JSX 語法

    摘要:需要提醒讀者的是,的很多例子都是通過來寫的,但這并不是語法,后面我們會有單獨的一小節(jié)講解的基本語法,不過目前為止我們先將跟多精力放在上。 書籍完整目錄 1.2 JSX 語法 showImg(https://segmentfault.com/img/bVvKLR); 官方文檔 https://facebook.github.io/react/docs/jsx-in-depth.html ...

    moven_j 評論0 收藏0
  • 精益 React 學(xué)習(xí)指南Lean React)- 1.1 React 介紹

    摘要:單向數(shù)據(jù)流應(yīng)用的核心設(shè)計模式,數(shù)據(jù)流向自頂向下我也是性子急的人,按照技術(shù)界的慣例,在學(xué)習(xí)一個技術(shù)前,首先得說一句。然而的單向數(shù)據(jù)流的設(shè)計讓前端定位變得簡單,頁面的和數(shù)據(jù)的對應(yīng)是唯一的我們可以通過定位數(shù)據(jù)變化就可以定位頁面展現(xiàn)問題。 書籍完整目錄 1.1 React 介紹 showImg(https://segmentfault.com/img/bVvJgS); 1.1.1 React ...

    lsxiao 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<