ReactNative中的高階組件(HOC)和繼承詳解

ReactNative中的高階組件(HOC)和繼承詳解

共同點:

  1. 高階組件(HOC)是 React 中用于復(fù)用組件邏輯的一種高級技巧册烈。HOC 自身不是 React API 的一部分删性,它是一種基于 React 的組合特性而形成的設(shè)計模式。
  2. 繼承也是用于解決復(fù)用代碼的一種技巧

不同點:

  1. 高階組件是參數(shù)為組件钓辆,返回值為新組件的函數(shù)
  2. 繼承是類繼承類坤按,是面向?qū)ο蟮囊淮筇攸c

本文以FlatList為例惊豺,詳解高階組件和繼承的使用!為啥是FlatList尼坚俗?
因為FlatList使用廣泛镜盯,造成大量重復(fù)代碼,僅僅是因為樣式不同而導(dǎo)致猖败,使用高階組件速缆,可以傳入被包裹的樣式組件,由高階組件統(tǒng)一實現(xiàn)FlatList的方法恩闻,由此可以減少重復(fù)代碼艺糜,減少耦合,其他場景等同

高階組件(HOC)

高階組件的樣式:

function HOC(WrappedComponent) {
    return class Component extends React.Component {
    render() {
      // 將WrappedComponent組件包裝在容器中幢尚,而不對其進行修改破停。Good!
      return <WrappedComponent {...this.props} />;
    }
  }

而我們要改變的不是傳入的FlatList,而是renderItem組件尉剩,首先在高階組件中實現(xiàn)FlatList:

 keyExtractor = (item, index) => {
            return index.toString()
        }

        listHeaderComponent = () => {
            return (
                <View style={{marginTop: Dimens.dx_60}}/>
            )
        }

        listFooterComponent = () => {
            if (this.state.showFoot) {
                return (
                    <View style={styles.footView}>
                        <Text style={styles.footText}>
                            沒有更多數(shù)據(jù)了
                        </Text>
                    </View>
                )
            } else {
                return (
                    <View style={styles.footLoadingView}>
                        <Text style={styles.footLoadingText}>
                            正在加載中...
                        </Text>
                    </View>
                )
            }
        }
        onEndReached = () => {
            if (this.hasNextPage === 1) {
                this.pageIndex++,
                    this.setState({
                        showFoot: false
                    })
                this.getNetworkingDataFunction(this.pageIndex, 20)
            } else {
                this.setState({
                    showFoot: true
                })
            }
        }

        flatListRenderItem = (item) => {
            return (
                <WrappedComponent {...this.props} {...item} />
            )
        }

        onRetry = () => {
            this.getNetworkingDataFunction(this.pageIndex, 20)
        }

        render() {
            if (this.state.error) {
                return (
                    <LoadFailView onRetry={this.onRetry}/>
                )
            }
            if (this.state.list && this.state.list.length) {
                return (
                    <View style={{backgroundColor: Colors.bgColor, flex: 1}}>
                        <FlatList
                            keyExtractor={this.keyExtractor.bind(this)}
                            horizontal={false}
                            data={this.state.list}
                            numColumns={WrappedComponent.numOfColumn()}
                            showsVerticalScrollIndicator={false}
                            renderItem={this.flatListRenderItem.bind(this)}
                            ListHeaderComponent={this.listHeaderComponent.bind(this)}
                            ListFooterComponent={this.listFooterComponent.bind(this)}
                            onEndReached={this.onEndReached.bind(this)}
                            onEndReachedThreshold={0.01}
                        />
                    </View>
                )
            } else {
                return <EmptyView text={this.state.empty}/>
            }

        }
    }

毫無疑問在以上代碼中你已經(jīng)看到flatListRenderItem返回的組件才是被包裹組件真慢,現(xiàn)在只需要解決數(shù)據(jù)問題,高階組件FlatList就完成了

由于高階組件中獲取不了被包裹組件的實例理茎,需要在被包裹組件中聲明靜態(tài)網(wǎng)絡(luò)請求方法:getNetworkingFunction

/**
   * Takes network data from server
   * 無默認(rèn)配置黑界,組件中有網(wǎng)絡(luò)請求需實現(xiàn)異步Promise:
   **/
getNetworkingDataFunction = (pageIndex, pageSize) => {
   WrappedComponent.getNetworkingFunction(pageIndex, pageSize).then(results => {

            }).catch(error => {
                
                            })
}

在被包裹組件中:

此處需要用到異步Promise

static getNetworkingFunction = (pageIndex,pageSize)=> {
       return new Promise(function (resolve, reject) {
           requestGetIntegralExchangeRecordList(pageIndex, pageSize).then(results => {
               resolve(results)
           }).catch(error => {
               reject(error)
           })
       })
    }

用來控制FlatList的column

static numOfColumn = () =>{
      return 1;
    }
    
傳入各樣式的item組件即可
    
render() {

}

繼承

父類:

export default class BaseComponent extends React.Component {

}

子類:
export default class childComponent extends BaseComponent {

}

父類中實現(xiàn)FlatList

/**
     * Takes network data from server
     * 無默認(rèn)配置,必須繼承實現(xiàn)
     **/
    getNetworkingDataFunction = () =>{

    }

    /**
     * Takes an item from data and renders it into the list. Typical usage:
     * 無默認(rèn)配置皂林,必須繼承實現(xiàn)
     */
    flatListRenderItem = (item) =>{

    }

    /**
     * Rendered at the very beginning of the list.
     * 默認(rèn)配置如下朗鸠,可通過繼承實現(xiàn)修改
     */
    listHeaderComponent = () => {
        return (
            <View style={{marginTop: Dimens.dx_60}}/>
        )
    }
    /**
     * Rendered at the very end of the list.
     * 默認(rèn)配置如下,可通過繼承實現(xiàn)修改
     */
    listFooterComponent = () => {
        if (this.state.showFoot) {
            return (
                <View style={styles.footView}>
                    <Text style={styles.footText}>
                        沒有更多數(shù)據(jù)了
                    </Text>
                </View>
            )
        } else {
            return (
                <View style={styles.footLoadingView}>
                    <Text style={styles.footLoadingText}>
                        正在加載中...
                    </Text>
                </View>
            )
        }
    }

    keyExtractor = (item, index) => {
        return index.toString()
    }

    onEndReached = () =>{
        if (this.hasNextPage === 1) {
            this.pageIndex++,
                this.setState({
                    showFoot: false
                })
            this.getNetworkingDataFunction(this.pageIndex,20)
        } else {
            this.setState({
                showFoot: true
            })
        }
    }

    render(){
        if (this.state.list && this.state.list.length){
            return (
                <View style={{backgroundColor: Colors.bgColor, flex: 1}}>
                    <FlatList
                        keyExtractor={this.keyExtractor.bind(this)}
                        horizontal={false}
                        data={this.state.list}
                        showsVerticalScrollIndicator={false}
                        renderItem={this.flatListRenderItem.bind(this)}
                        ListHeaderComponent={this.listHeaderComponent.bind(this)}
                        ListFooterComponent={this.listFooterComponent.bind(this)}
                        onEndReached={this.onEndReached.bind(this)}
                        onEndReachedThreshold={0.01}
                    />
                </View>
            )
        }else {
            return <EmptyView text={this.state.emptyText}/>
        }

    }

在子類中去重寫父類的方法


* 網(wǎng)絡(luò)請求方法:getNetworkingDataFunction
* render內(nèi)容展示:flatListRenderItem

getNetworkingDataFunction = (pageIndex,pageSize) => {

        requestGetIntegralExchangeRecordList(pageIndex, pageSize).then(results => {
        
        }).catch(error => {

        })
    }

flatListRenderItem = ({item}) =>{
        return (
            <TouchableOpacity onPress={event => {
                this.onClick(item)
            }}>
                <RenderItem item={item} />
            </TouchableOpacity>
        )
    }

onClick(item) {
 }
 
class RenderItem extends React.Component{

}

那為什么react-native不推薦使用繼承础倍?
react中推薦使用組合和高階組件來替代繼承童社,網(wǎng)上關(guān)于react的繼承也是微乎其微,對于iOS出身的我來說著隆,繼承主要有一下缺點:

  1. 父類的內(nèi)部細(xì)節(jié)對子類是可見的
  2. 如果父類做了修改扰楼,所有涉及的子類必須修改呀癣,高耦合。
  3. 繼承也有可能使程序變得難以閱讀弦赖。調(diào)用一個方法時,有時很難判斷它是在哪定義的项栏,子類中查找問題必須去父類中查看
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市蹬竖,隨后出現(xiàn)的幾起案子沼沈,更是在濱河造成了極大的恐慌,老刑警劉巖币厕,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件列另,死亡現(xiàn)場離奇詭異,居然都是意外死亡旦装,警方通過查閱死者的電腦和手機页衙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來阴绢,“玉大人店乐,你說我怎么就攤上這事∩胂” “怎么了眨八?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長左电。 經(jīng)常有香客問我廉侧,道長,這世上最難降的妖魔是什么篓足? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任段誊,我火速辦了婚禮,結(jié)果婚禮上纷纫,老公的妹妹穿的比我還像新娘枕扫。我一直安慰自己,他們只是感情好辱魁,可當(dāng)我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布烟瞧。 她就那樣靜靜地躺著,像睡著了一般染簇。 火紅的嫁衣襯著肌膚如雪参滴。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天锻弓,我揣著相機與錄音砾赔,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛暴心,可吹牛的內(nèi)容都是我干的妓盲。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼专普,長吁一口氣:“原來是場噩夢啊……” “哼悯衬!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起檀夹,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤筋粗,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后炸渡,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體娜亿,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年蚌堵,在試婚紗的時候發(fā)現(xiàn)自己被綠了买决。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡辰斋,死狀恐怖策州,靈堂內(nèi)的尸體忽然破棺而出瘸味,到底是詐尸還是另有隱情宫仗,我是刑警寧澤,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布旁仿,位于F島的核電站藕夫,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏枯冈。R本人自食惡果不足惜毅贮,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望尘奏。 院中可真熱鬧滩褥,春花似錦、人聲如沸炫加。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽俗孝。三九已至酒甸,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間赋铝,已是汗流浹背插勤。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人农尖。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓析恋,卻偏偏與公主長得像,于是被迫代替她去往敵國和親盛卡。 傳聞我的和親對象是個殘疾皇子绿满,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容

  • 前言 學(xué)習(xí)react已經(jīng)有一段時間了嚎货,期間在閱讀官方文檔的基礎(chǔ)上也看了不少文章橘霎,但感覺對很多東西的理解還是不夠深刻...
    Srtian閱讀 1,657評論 0 7
  • 本文章將從四個方面講解高階組件 什么是高階組件? 高階組件是為了解決什么問題殖属? 常見用法 與父組件的區(qū)別 什么是高...
    rangel閱讀 3,297評論 0 3
  • 原文地址:https://github.com/SmallStoneSK/Blog/issues/6 1. 前言 ...
    小石頭若海閱讀 916評論 2 2
  • 高階組件是react應(yīng)用中很重要的一部分姐叁,最大的特點就是重用組件邏輯。它并不是由React API定義出來的功能洗显,...
    叫我蘇軾好嗎閱讀 900評論 0 0
  • React進階之高階組件 前言 本文代碼淺顯易懂外潜,思想深入實用。此屬于react進階用法挠唆,如果你還不了解react...
    流動碼文閱讀 1,185評論 0 1