状态(state)

状态就是组件描述某种显示情况的数据,由组件自己设置和更改,也就是所组件自己维护,目的就是为了在不同状态下使组件的显示不同

在组件中只能通过getInitialState的钩子函数来给组件挂载初始状态,在组件内部通过this.state获取

this.props和this.state是纯js对象,在vue中,$data属性是利用Object.defineProperty处理过的,更改$data的数据的时候会触发数据的getter和setter,但是React中没有做这样的处理,如果直接更改的话,React是无法得知的,所以需要使用特殊的更改状态的方法

setState(params)

在setState中传入一个对象,就会将组件的状态中键值对的部分更改,还可以传入一个函数,这个回调函数必须返回像上面方式一样的一个对象,函数可以接收prevState和props

实现下拉菜单的方式

1. 通过数据来控制元素的行内样式中display

<ul style={{display:isMenuShow?'block':'none'}}><li>国内新闻</li></ul>
<ul className={isMenuShow?'show':'hide'}><li>国内新闻</li></ul>

2. 根据数据控制是否渲染该节点、组件

{
isMenuShow?<ul><li>国内新闻</li></ul>:''
}

3. 通过ref对dom、组件进行标记,在组件内部通过this.refs获取到之后,进行操作

<ul ref='content'><li>国内新闻</li></ul>this.refs.content.style.display = this.state.isMenuShow?'block':'none'

属性和状态的对比

相似点:都是纯js对象,都会触发render更新,都具有确定性(状态/属性相同,结果相同)

不同点:

1. 属性能从父组件获取,状态不能

2. 属性可以有父组件修改,状态不能

3. 属性能在内部设置默认值,状态不可以

4. 属性不在组件内部修改,状态要改

5. 属性能设置子组件初始值,装填不可以

6. 属性可以修改子组件的值,状态不可以

状态只和自己相关,由自己维护

属性不要自己修改,可以从父组件获取,也可以给子组件设置

组件在运行时自己需要修改的数据其实就是状态而已

父子组件传参的交互

我们经常会令父组件将自己的某一条状态传递给子组件,这个时候,当父组件的状态更改的时候,子组件接收到的属性就会发送改变,触发render重新执行

也就是说state更改的时候(使用setState),会触发render重新执行,属性更改的时候,也能做到这一点

组件的生命周期

组件是一个构造器,每一次使用组件都相当于在实例化组件,这个时候,组件就会经历一次生命周期,从实例化实例开始到这个实例销毁的时候,都是一个完整的生命周期

组件的生命周期,我们会分为三个阶段,初始化、运行中、销毁

初始化阶段

1. 实例化组件之后,组件的getDefaultProps钩子函数会执行

这个钩子函数的目的是为组件的实例挂载默认的属性

这个钩子函数只会执行一次,也就是说,只在第一次实例化的时候执行,创建出所有实例共享的默认属性,后面再实例化的时候,不会执行getDefaultProps,直接使用已有的共享的默认属性

理论上来说,写成函数返回对象的方式,是为了防止实例共享,但是React专门为了让实例共享,只能让这个函数只执行一次

组件间共享默认属性会减少内存空间的浪费,而且也不需要担心某一个实例更改属性后其他的实例也会更改的问题,因为组件不能自己更改属性,,而且默认属性的优先级低

2. 执行getInitialState为实例挂载初始状态,且每次实例化都会执行,也就是说,每一个组件实例都拥有自己独立的状态

3. 执行componentWillMount,相当于vue里的created+beforeMount,这里是在渲染前最后一次更改数据的机会,在这里更改不会触发render的重新执行,在这里做数据的获取

4. 执行render,渲染dom

5. 执行componentDidMount,相当于vue里的mouted,多用于操作真实dom

运行中阶段

当组件mount到页面中之后,就进入了运行中阶段,在这有5个钩子函数,但这5个钩子函数只有数据(属性、状态)发生改变的时候才会执行

1. componentWillReceiveProps

当父组件给子组件传入的属性改变的时候,子组件的这个函数才会执行

当执行的时候,函数接收的第一个参数是子组件接收到的新参数,这个时候,新参数还没有同步到this.props上,多用于判断新属性和原有属性的变化后更改组件的状态

2. 接下来会执行shouldComponentUpdated,这个函数的作用

当属性或状态发送改变后控制组件是否更新,提高性能,返回true就更新,否则不更新,默认返回true

接收nextProp、nextState,根据新属性状态和原属性状态进行对比,判断后控制是否更新

3. componentWillUpdate,在这里,组件马上就要重新render了,多做一些准备工作,千万不要在这里修改状态,否则会死循环

相当于vue中的beforeUpdate

4. render,重新渲染dom,不能修改状态,会死循环

5. componentDidUpdate,在这里新的dom结构产生,不能修改状态,会死循环

销毁阶段

当组件被销毁之前,会触发componentWillUnmount

相当于vue里的beforeDestroy,所以会执行一些事件

为什么vue中有beforeDestroy,而React却没有?

vue在调用$destroy方法的时候就会执行beforeDestroy,然后组件被销毁,这个时候组件的结构还存在于页面结构中,也就是说如果想要对残留的dom结构进行处理必须在destroyed处理,但是React执行完componentWillUNmount之后吧事件、数据、dom都全部处理掉了,所以根本不需要其他的钩子函数了

怎么样就算组件被销毁:

1. 当父组件从渲染这个子组件变成不渲染这个子组件的时候,子组件相当于被销毁

2. 调用ReactDOM.unmountComponentAtNode(node)方法来将某节点中的组件销毁

react中的事件对象

React中对于事件进行了处理,解决了一些兼容性问题,React事件对象上面挂载着nativeEvent,这个就是原生的事件对象

React对事件对象做了优化,如果不取值的话,只都是null

React中组件通信方式

父组件与子组件通信

1. 父组件将自己的状态传递给子组件,子组件当做属性来接收,当父组件更改自己状态的时候,子组件接收到的属性就会发生改变

2. 父组件利用ref对子组件做标记,通过调用子组件的方法以更改子组件的状态

子组件与父组件通信

1. 父组件将自己的某个方法传递给子组件,在方法里可以做任意操作,比如可以更改状态,子组件通过this.props接收到父组件的方法后调用

兄弟组件通信

在React没有类似vue中的事件总线来解决这个问题,我们只能借助他们共同的父级组件来实现,将非父子关系转换成多维度的父子关系

复杂的非父子组件通信在React中很难处理,多组件间的数据共享也不好处理,所以我们会使用flux、redux来实现这样的功能,解决这个问题

React中表单元素默认值

在React中,如果需要给表单元素设置默认value和checked,需要设置成defaultValue/defaultChecked,否则设置默认之后,用户无法更改

React中的mixins

在vue中我们可以将一些通用的、公用的方法放入到某一个纯js对象中,然后,在需要使用该方法的组件中使用mixins配置(值为对象)将该js对象中的方法注入到组件中,这样就能实现代码复用,便于维护

在React中曾经也有这样的api,但是在高版本React中推荐我们使用es6中的class来创建组件了,这个时候无法使用mixinsapi,所以mixins被废弃了,如果要使用公用代码抽离,我们可以使用模块化

React-keys

我们在React中循环列表数据的时候,需要对循环出来的虚拟jsx节点传入key这个数据

keys可以在DOM中的某些元素被增加或删除的时候帮助React识别哪些元素发生了变化,因此你应该给数组中的每一个元素赋予一个确定的标识

状态提升

如果有多个组件共享一个数据,把这个数据放到共同的父级组件中来管理

组合

在vue中有一个内容分发叫slot,在React中也有实现,就是可以在使用组件的时候,在组件标签内部放入一些不固定的内容,在该组件的模板中,只有{this.props.children}来表示

//App
<Dialog
close={this.ToggleDialogShow} isShow={isDialogShow}
>
<ContentA/>
<ContentA/>
<ContentB/>
</Dialog> //dialog
<div style={{display:isShow?'block':'none'}} className="dialog">
<Button handler={this.props.close} text="关闭"/> {this.props.children}//这里就是slot </div>

React学习之路(二)的更多相关文章

  1. Redis——学习之路二(初识redis服务器命令)

    上一章我们已经知道了如果启动redis服务器,现在我们来学习一下,以及如何用客户端连接服务器.接下来我们来学习一下查看操作服务器的命令. 服务器命令: 1.info——当前redis服务器信息   s ...

  2. react.js学习之路二

    看react.js对我来说真的不是难不难的问题,问题是我的思路太难转变了,真是坑死我了,react里面的坑也很多.算了,这些抱怨没啥用,记录一下今天学习的内容. 今天看了to-do-list经典示例 ...

  3. zigbee学习之路(二)点亮LED

    一.前言 今天,我来教大家如何点亮led,这也是学习开发板最基础的步骤了. 二.原理分析 cc2530芯片跟虽然是51的内核,但是它跟51单片机还是有区别的,51单片机不需要对IO口进行配置,而cc2 ...

  4. React 学习之路 (一)

    先说一说对React的体验,总结 首先react相对angular来说入手简单暴力,在学习的这段时间里发现: 我们每天做的事就是在虚拟DOM上创建元素然后在渲染到真实的DOM中 渲染到真实DOM上的R ...

  5. react学习笔记(二)编写第一个react组件

    继续上一节课的内容,打开App.js:会看到如下代码: import React, { Component } from 'react';  //在此文件中引用React,以及reat的组件类 imp ...

  6. Android开发学习之路-二维码学习

    这个月装逼有点少了,为什么呢,因为去考软件射鸡师了,快到儿童节了,赶紧写篇博纪念一下逝去的青春,唔,请忽略这句话. 二维码其实有很多种,但是我们常见的微信使用的是一种叫做QRCode的二维码,像下面这 ...

  7. Python学习之路二

    今天主要学习了列表,python的列表真的事太强大了,由于内容比较多,今天就先简单的介绍一下新学的几个成员函数吧. 首先我们要了解list是一种序列类型,其构造方式有四种形式: (1)空列表 [] ( ...

  8. React学习小结(二)

    一.组件的嵌套 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <t ...

  9. Spring学习之路二——概念上理解Spring

    一.概念. Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson 在其著作Expert One-On-One J2EE Develop ...

随机推荐

  1. 浏览器抓包(post)

    谁能想到我写的第一个wp竟然是个web题??? 以Geek2017_Buy me a Tesla 为例 来看看题目 emmmm有理想还是很好的,火狐打开(事实证明FF对web题是最友好的) 能点的地方 ...

  2. MongoDB之建立Windows和本地虚拟机的双向连接

    本文主要分享如何将MongoDB数据库在Windows系统和本地虚拟机系统建立双向连接,我们将借助MongoDB的可视化工具Robomongo来实现.首先,应该确保你的Windows系统和本地虚拟机系 ...

  3. (win10 64位系统中)Visual Studio 2015+OpenCV 3.3.0环境搭建,100%成功

    (win10 64位系统中)Visual Studio 2015+OpenCV 3.3.0环境搭建,100%成功 1.下载opencv 官网http://opencv.org/下载windows版Op ...

  4. 解决failed to push some refs to git

    Administrator@PC-20150110FGWU /K/cocos2d/yc (master) $ git push -u origin master To git@github.com:y ...

  5. Java-----关于eclipse导入项目发生的问题及解决办法

    今天通过eclipse导入了几个项目,项目名出现红叉,对于我这样的强迫症来说是无法容忍的,故现做总结,遇到同学可按照以下方法来操作. 改动的地方主要是两个方面: 1.Tomcat版本问题. 此问题是由 ...

  6. vue项目的骨架及常用组件介绍

    vue项目基础结构 一个vue的项目,我觉得最小的子集其实就是{vue,vue-router,component},vue作为基础库,为我们提供双向绑定等功能.vue-router连接不同的" ...

  7. Oracle 11gR2 用exp无法导出空表解决方法

    Oracle 11gR2 用exp无法导出空表解决方法 在11gR2中有个新特性,当表无数据时,不分配segment以节省空间.Oracle 当然在运行export导出时,空表则无法导出,可是还是有解 ...

  8. 【蓝牙低功耗BLE】控制GPIO来点亮LED

    这节讲一下最简单的,也是最基础的东西.CC2540的IO操作,把PORT口当做GPIO来用,废话不多说,往下看. 1.硬件电路 硬件电路时最简单的,用一根GPIO去控制LED灯.因为GPIO作为out ...

  9. MVC系列——一个异常消息传递引发的思考

    前言:最近在某个项目里面遇到一个有点纠结的小问题,经过半天时间的思索和尝试,问题得到解决.在此记录一下解决的过程,以及解决问题的过程中对.net里面MVC异常处理的思考.都是些老生常谈的问题,不多说, ...

  10. SpringMVC框架(三)from标签(转)

    原文地址:http://blog.csdn.net/kutim/article/details/46682547 spring表单标签   <%@taglib uri="http:// ...