ReactNative中的高阶组件(HOC)和继承详解

ReactNative中的高阶组件(HOC)和继承详解

共同点:

  1. 高阶组件(HOC)是 React 中用于复用组件逻辑的一种高级技巧。HOC 自身不是 React API 的一部分,它是一种基于 React 的组合特性而形成的设计模式。
  2. 继承也是用于解决复用代码的一种技巧

不同点:

  1. 高阶组件是参数为组件,返回值为新组件的函数
  2. 继承是类继承类,是面向对象的一大特点

本文以FlatList为例,详解高阶组件和继承的使用!为啥是FlatList尼?
因为FlatList使用广泛,造成大量重复代码,仅仅是因为样式不同而导致,使用高阶组件,可以传入被包裹的样式组件,由高阶组件统一实现FlatList的方法,由此可以减少重复代码,减少耦合,其他场景等同

高阶组件(HOC)

高阶组件的样式:

function HOC(WrappedComponent) {
    return class Component extends React.Component {
    render() {
      // 将WrappedComponent组件包装在容器中,而不对其进行修改。Good!
      return <WrappedComponent {...this.props} />;
    }
  }

而我们要改变的不是传入的FlatList,而是renderItem组件,首先在高阶组件中实现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}>
                            没有更多数据了
                        </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}/>
            }

        }
    }

毫无疑问在以上代码中你已经看到flatListRenderItem返回的组件才是被包裹组件,现在只需要解决数据问题,高阶组件FlatList就完成了

由于高阶组件中获取不了被包裹组件的实例,需要在被包裹组件中声明静态网络请求方法:getNetworkingFunction

/**
   * Takes network data from server
   * 无默认配置,组件中有网络请求需实现异步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 {

}

父类中实现FlatList

/**
     * Takes network data from server
     * 无默认配置,必须继承实现
     **/
    getNetworkingDataFunction = () =>{

    }

    /**
     * Takes an item from data and renders it into the list. Typical usage:
     * 无默认配置,必须继承实现
     */
    flatListRenderItem = (item) =>{

    }

    /**
     * Rendered at the very beginning of the list.
     * 默认配置如下,可通过继承实现修改
     */
    listHeaderComponent = () => {
        return (
            <View style={{marginTop: Dimens.dx_60}}/>
        )
    }
    /**
     * Rendered at the very end of the list.
     * 默认配置如下,可通过继承实现修改
     */
    listFooterComponent = () => {
        if (this.state.showFoot) {
            return (
                <View style={styles.footView}>
                    <Text style={styles.footText}>
                        没有更多数据了
                    </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}/>
        }

    }

在子类中去重写父类的方法


* 网络请求方法:getNetworkingDataFunction
* render内容展示: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中推荐使用组合和高阶组件来替代继承,网上关于react的继承也是微乎其微,对于iOS出身的我来说,继承主要有一下缺点:

  1. 父类的内部细节对子类是可见的
  2. 如果父类做了修改,所有涉及的子类必须修改,高耦合。
  3. 继承也有可能使程序变得难以阅读。调用一个方法时,有时很难判断它是在哪定义的,子类中查找问题必须去父类中查看
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容

  • 前言 学习react已经有一段时间了,期间在阅读官方文档的基础上也看了不少文章,但感觉对很多东西的理解还是不够深刻...
    Srtian阅读 1,647评论 0 7
  • 本文章将从四个方面讲解高阶组件 什么是高阶组件? 高阶组件是为了解决什么问题? 常见用法 与父组件的区别 什么是高...
    rangel阅读 3,277评论 0 3
  • 原文地址:https://github.com/SmallStoneSK/Blog/issues/6 1. 前言 ...
    小石头若海阅读 910评论 2 2
  • 高阶组件是react应用中很重要的一部分,最大的特点就是重用组件逻辑。它并不是由React API定义出来的功能,...
    叫我苏轼好吗阅读 888评论 0 0
  • React进阶之高阶组件 前言 本文代码浅显易懂,思想深入实用。此属于react进阶用法,如果你还不了解react...
    流动码文阅读 1,183评论 0 1