深入React组件的数据:prop、state

  • React组件的数据分为两种,propstate,无论prop或者state改变,都可能引发组件的重新渲染。
  • prop是组件对外接口,state是组件的内部状态,对外用prop,对内用state
    (如果你熟悉VueReact中的prop就相当于Vue组件的data() { return {} }里的数据,Reactstate就相当于Vue中的prop
  • 组件不应该改变prop的值,而state存在的目的就是让组件来改变。

一、prop

React中,prop(propety的简写)是从外部传递给组件的数据,一个组件通过定义自己能够接受的prop就定义了自己对外公共接口。

1.给prop赋值
// 一段JSX代码中:
<SampleButton id="sample" borderWidth={2} onClick={onButtonClick} style={{ color: "red" }} />
  • 在例子中,创建了名字为SampleButton 的组件实例,idborderWidthonClickstyle四个prop
  • HTML中的属性,只支持字符串类型,而React组件,除了支持字符串,还支持数字(2),函数(onButtonClick ),对象({ color: "red" })。
  • 只要prop值不是字符串类型,就必须要用{}prop值括住。所以对象变量会有两层{{}}
2.读取prop的值
class Counter extends Component {
  constructor(props) {
    super(props)

    this.onClickIncrementButton = this.onClickIncrementButton.bind(this)
    this.onClickDecrementButton = this.onClickDecrementButton.bind(this)

    this.state = {
      count: props.initValue || 0    
    }
  }
}
  • 如果一个组件需要定义自己的构造函数,一定要记得在构造函数的第一行通过super调用父类也就是React.Component 的构造函数。

  • 如果在构造函数中没用调用super(props),那么组件实例被构造之后,类实例的所有成员就无法通过this.props 访问到父组件传递过来的props值。

  • 很明显,给this.props 赋值是React.Component 构造函数的工作之一。

  • Counter的构造函数中还给两个成员版定了当前this的执行环境,因为ES6的构造方法创造的React组件类并不自动给我们绑定this到当前实例对象。

  • 在构造函数中,可以通过参数props获得传入的prop值。

  • 其他函数中,通过this.props获得传入的prop值,还可以使用ES6的结构赋值:(const { test } = this.props 相当于const test = this.props.test

3.propTypes检查

ES6方法定义的组件类中,可以通过增加类的propTypes属性来定义prop的规格,这不只是声明,还是一种限制。在运行时和静态代码检查时,都可以根据propTypes 判断外部世界是否正确地使用了组件。

Counter.propTypes = {
  caption: PropTypes.string.isRequiered,
  initValue: PropTypes.number
}
  • 其中要求 caption必须是 string 类型
  • 其中要求 initValue 必须是 number 类型
  • 其中要求 caption 必需

propTypes虽然能够在开发阶段发现代码中的问题,但是放在产品环境不合适,会占用一些代码空间,消耗CPU计算资源。所以在开发阶段使用,当部署到环境产品时应该去掉。
babel-react-optimize就有这个功能,通过npm来安装,但是应该确保只在发布产品代码时使用它。

二、state

state表示组件的内部状态。由于React组件不能修改传入的prop,所以需要记录自身数据变化,就要用到state

1.初始化state

通常都会在组件类的构造函数的结尾处初始化state

constructor(props) {
  ...
  this.state = {
    count: props.initValue || 0
  }
}
  • 通过对this.state的赋值完成了对组件state的初始化
  • 组件的state,必须是一个JavaScript对象
  • 如果在propTypes声明中有用isRequired要求必须有值的prop,则需要在代码中判断所给prop值是否存在,如果不存在,就给一个默认的初始值。

可以用ReactdefaultProps

Counter.defaultProps = {
  initValue: 0
}

这样,Counter构造函数中的this.state初始化中可以省去判断条件,代码可以简化为:

constructor(props) {
 ...
 this.state = {
   count: props.initValue
 }
}

如果实例中即使没有指定count属性的initValue,实例组件中的count属性也能收到一个默认的属性值0。

2.读取和更新state

通过给buttononClick属性挂载点击事件处理函数,我们可以改变组件的state,以点击按钮给count加1的响应事件为例:

onClickIncrementButton() {
  this.setState({ count: this.state.count + 1 })
}
  • 在事件中,由于绑定了this,就能通过this.state读取到组件的当前state
  • 改变组件state必须要试用this.setState函数,而不能直接去修改this.state
  • 直接修改this.state值,虽然改变了组件的内部状态,但不会驱动组件进行重新渲染,就不会实时反应this.state的值的变化。
  • this.setState函数所做的事情,首先是改变this.state的值,然后驱动组件经历更新过程,这样this.state才会以新的值出现在界面上。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352