4-React 组件之状态与通信

React.js

[TOC]

组件

web 组件就是对 web 中的数据、结构、方法等进行封装,复用,与 JavaScript 中功能函数封装类似,但更关注的是对 web 元素(标签)的封装与扩展(扩展:webComponent

React 提供了两种组件构建方式

  • 函数式组件
  • 类式组件

函数式组件

在 <u>React.js</u> 中,定义一个组件的最简单的方式就是 <u>函数</u>

function Kaikeba() {
    return (
        <div>
            <h2>开课吧!</h2>
        </div>
    );
}
ReactDOM.render(
    <Kaikeba />,
    document.getElementById('app')
);
  • 函数的名称就是组件的名称
  • 函数的返回值就是组件要渲染的内容

<span style="color:red">函数的名称(组件的名称),必须是首字母大写</span>

类式组件

我们还可以通过 <u>类(class)</u>类定义组件

class Miaov extends React.Component {
    render() {
        return (
            <div>
                <h2>妙味!</h2>
            </div>
        );
    }
}
  • 组件类必须继承 <u>React.Component</u>
  • 组件类必须有 <u>render</u> 方法
  • <u>render</u> 方法的返回值就是组件要渲染的内容

<span style="color:red">类的名称(组件的名称),也必须是首字母大写</span>

不从实际案例开始的知识点就是耍 LM

[图片上传失败...(image-12e34-1613962628611)]

样式

.friend-list {
    border: 1px solid #000000;
    width: 200px;
}
.friend-group dt {
    padding: 10px;
    background-color: rgb(64, 158, 255);
    font-weight: bold;
}
.friend-group dd {
    padding: 10px;
    display: none;
}
.friend-group.expanded dd {
    display: block;
}
.friend-group dd.checked {
    background: green;
}

创建 <u>FriendList</u> 组件

class FriendList extends React.Component {
    render() {
        return (
            <div className="friend-list">
                <div className="friend-group">
                    <dt>家人</dt>
                    <dd>爸爸</dd>
                    <dd>妈妈</dd>
                </div>
                <div className="friend-group">
                    <dt>朋友</dt>
                    <dd>张三</dd>
                    <dd>李四</dd>
                    <dd>王五</dd>
                </div>
                <div className="friend-group">
                    <dt>客户</dt>
                    <dd>阿里</dd>
                    <dd>腾讯</dd>
                    <dd>头条</dd>
                </div>
            </div>
        );
    }
}

组件复用 - 数据抽取

为了提高组件的复用性,通常会把组件中的一些可变数据提取出来

let datas = {
    family: {
        title: '家人',
        list: [
            {name: '爸爸'},
            {name: '妈妈'}
        ]
    },
    friend: {
        title: '朋友',
        list: [
            {name: '张三'},
            {name: '李四'},
            {name: '王五'}
        ]
    },
    customer: {
        title: '客户',
        list: [
            {name: '阿里'},
            {name: '腾讯'},
            {name: '头条'}
        ]
    }
};

组件复用 - 数据传入

数据虽然分离了,但是为了降低组件与数据之前的依赖,我们应该尽量避免在组件内部直接访问数据,通过传参的方法来进行解耦

property

使用组件的时候通过 <u>标签属性-property</u> 的方式传入数据,在组件内部通过构造函数参数(如果是函数组件,则通过函数参数)来接收传入的数据

<组件名称 属性名称="值" />
// 使用表达式
<组件名称 属性名称={表达式} />
ReactDOM.render(
    <FriendList datas={datas} />,
    document.getElementById('app')
);

接收参数 - props

在组件对应的函数或类中通过:

  • 函数式组件:通过函数的第一个参数来接收
  • 类式组件:通过类的构造函数第一个参数来接收

无论是函数式组件还是类式组件,都会把传入的参数组装成一个对象:

<组件名称 属性名称1="值1" 属性名称二={表达式二} />

// 函数式组件
function 组件名称(参数) {
    // 参数的结构为
    参数 = {
        属性名称1: "值1",
        属性名称二: 表达式二的值
    }
  
      return <div>组件结构</div>
}
// 类式组件
class 组件名称 extends React.Component {
    constructor(参数) {
        super(参数);
      
        this.props = {
            属性名称1: "值1",
            属性名称2: 表达式二的值
        }
    }
  
    render() {
        return <div>组件结构</div>
    }
}

在类式组件中,需要注意:

  • 当子类重写 <u>constructor</u> ,则必须调用父类 <u>super</u> - 类的知识点
  • 把接收到的参数传入父类构造函数,父类构造函数中会创建一个对象属性:<u>props</u> 类存储传入的参数数据

通过参数动态渲染组件结构

class FriendList extends React.Component {
    constructor(props) {
        super(props);
    }
  
    render() {
        let {datas} = this.props;
        return (
            <div className="friend-list">
                    {Object.keys(datas).map((key, index) => (
                    <div className="friend-group" key={key}>
                        <dt>{datas[key].title}</dt>
                                {datas[key].list.map((list, index) => <dd key={index}>{list}</dd>)}
                    </div>
                ))}
            </div>
        );
    }
}

子组件提取

组件与类一样,是一个不断提取的过程,当我们发现某个部分可复用或者结构复杂的时候,我们可以对它再次进行提取

class FriendGroup extends React.Component {
        constructor(props) {
                super(props);
    }
  
    render() {
        let {data} = this.props;
        return (
                <div className="friend-group">
                <dt>{data.title}</dt>
                {data.list.map((list, index) => {
                        <dd key={index}>{list}</dd>
                })}
            </div>
        );
    }
}

class FriendList extends React.Component {
    constructor(props) {
        super(props);
    }
  
    render() {
        let {datas} = this.props;
        return (
            <div className="friend-list">
                    {Object.keys(datas).map((key, index) => (
                    <FriendGroup data={datas[key]} key={key} />
                ))}
            </div>
        );
    }
}

组件状态

有的时候,一个组件除了可以接收外部传入的数据,还会有属于自己的内部私有数据,同时组件私有数据的更新还会导致组件的重新渲染,比如上面的例子中的每一个好友面板会有一个内部私有数据来控制面板的展开与收缩,我们又把这种数据称为组件状态

state

存储组件状态的对象

class FriendGroup extends React.Component {
        constructor(props) {
                super(props);
        this.state = {
            expanded: false
        };
    }
  
    render() {
        let {data} = this.props;
        let {expanded} = this.state;
        return (
                <div className={[
                "friend-group",
                expanded && "expanded"
            ].join(' ')}>
                <dt>{data.title}</dt>
                {data.list.map((list, index) => {
                        <dd key={index}>{list}</dd>
                })}
            </div>
        );
    }
}

组件交互

一个组件除了有结构、样式,有的时候还会有交互,如上面的例子:现在我们希望当用户点击组件面板 title 的时候,面板组件的展开收缩状态就会发生改变

绑定事件

class FriendGroup extends React.Component {
        constructor(props) {
                super(props);
        this.state = {
            expanded: false
        };
        this.expand = this.click.expand();
    }
  
    expand() {
        this.setState({
            expanded: !this.state.expanded
        });
    }
  
    render() {
        let {data} = this.props;
        let {expanded} = this.state;
        return (
                <div className={[
                "friend-group",
                expanded && "expanded"
            ].join(' ')}>
                <dt onClick={this.expand}>{data.title}</dt>
                {data.list.map((list, index) => {
                        <dd key={index}>{list}</dd>
                })}
            </div>
        );
    }
}

<u>React.js</u> 的事件绑定需要注意:

  • <span style="color:red">事件名称是驼峰命名的</span>

  • <span style="color:red">事件绑定函数的 this 指向</span>

    • <span style="color:red">通过 bind 改变 this 指向,为了能够在方法中调用组件对象的属性和其它方法,我们需要把 this 指向组件</span>

    • <span style="color:red">通过箭头函数处理 this</span>

获取事件对象

事件绑定函数的第一个参数是事件对象,通过事件对象可以获取到原生 DOM 对象

状态更新

当我们希望更新组件状态的时候,不要直接修改 <u>state</u> 的值,而是需要调用 <u>setState</u> 方法来进行更新

setState 方法

<u>setState</u> 方法由父类 <u>React.Component</u> 提供,当该方法被调用的时候,组件的 <u>state</u> 会被更新,同时会重新调用组件的 <u>render</u> 方法对组件进行渲染

<span style="color:red">不要直接修改 <u>state</u>,而是调用 <u>setState</u> 方法来更新,组件构造函数是唯一可以对 <u>state</u> 直接赋值(初始化)的位置</span>

this.setState({key:val})
更新异步

出于性能考虑,<u>setState</u> 方法的修改并不是立即生效的

// this.state.val = 0
this.setState({
    val: this.state.val + 1
});
// this.state.val 的值并不是 1
console.log(this.state.val);
更新合并

<u>React</u> 会把多个 <u>setState</u> 合并成一个调用

// this.state.val = 0
this.setState({
    val: this.state.val + 1
});
this.setState({ // 因为异步的问题,this.state.val 的值在这里还是0
    val: this.state.val + 1
});
Object.assign(
  previousState,
  {val: state.val + 1},
  {val: state.val + 1},
  ...
)
回调函数

如果,有多个 <u>setState</u> 的调用,后一次的修改如果需要依赖上一次修改的值,那么可以使用函数的方式

// this.state.val = 0
this.setState((state, props) => {
    return {val: state.val + 1}
});
this.setState((state, props) => {
    return {val: state.val + 1}
});

props 与 state 的区别

<u>state</u> 的主要作用是用于组件保存、控制、修改自己的可变状态,在组件内部进行初始化,也可以在组件内部进行修改,但是组件外部不能修改组件的 <u>state</u>

<u>props</u> 的主要作用是让使用该组件的父组件可以传入参数来配置该组件,它是外部传进来的配置参数,组件内部无法控制也无法修改

<u>state</u> 和 <u>props</u> 都可以决定组件的外观和显示状态。通常,<u>props</u> 做为不变数据或者初始化数据传递给组件,可变状态使用 <u>state</u>

能使用 <u>props</u> 就不要使用 <u>state</u>

无状态组件

没有 <u>state</u> 的组件,我们称为无状态组件,因为状态会带来复杂性,所以,通常我们推荐使用无状态组件,也鼓励编写无状态组建

函数式组件

函数式组件没有 <u>state</u> ,所以通常我们编写使用函数式组件来编写无状态组件

组件通信与数据流

在上面的案例中,我们实现了一个多选式的好友列表(每个独立的面板都有自己独立的状态,互不影响)。下面我们再来看另外一种情况:单选式的好友列表(多组列表同时只能有一个为展开状态,也就是多个面板之间的状态是互相影响的)

状态提升

通过分析可以发现,面板的展开状态不在是组件内部私有状态了,多个组件都会受到这个状态的影响,也就是它们共享了一个状态。为了能够让多个不同组件共享同一个状态,这个时候,我们把这个状态进行提升,交给这些组件最近的公共父组件进行管理

class FriendGroup extends React.Component {
        constructor(props) {
                super(props);
      
        this.changeExpand = this.changeExpand.bind(this);
    }
  
    changeExpand() {
        typeof this.props.onExpand === 'function' &&            this.props.onExpand(this.props.index);
    }
  
    render() {
        let {data, expandedIndex, index} = this.props;
        return (
                <div onClick={this.changeExpand} className={[
                "friend-group",
                expandedIndex === index && "expanded"
            ].join(' ')}>
                <dt>{data.title}</dt>
                {data.list.map((list, index) => {
                        <dd key={index}>{list}</dd>
                })}
            </div>
        );
    }
}

class FriendList extends React.Component {
    constructor(props) {
        super(props);
      
        this.state = {
            expandedIndex: 0
        }
      
        this.changeExpandedIndex = this.changeExpandedIndex.bind(this);
    }
  
    changeExpandedIndex(expandedIndex) {
        this.setState({
            expandedIndex
        });
    }
  
    render() {
        let {datas} = this.props;
        return (
            <div onExpand={this.changeExpandedIndex} className="friend-list">
                    {Object.keys(datas).map((key, index) => (
                    <FriendGroup data={datas[key]} index={index} expandedIndex={expandedIndex} key={key} />
                ))}
            </div>
        );
    }
}

数据流

在 <u>React.js</u> 中,数据是从上自下流动(传递)的,也就是一个父组件可以把它的 <u>state / props</u> 通过 <u>props</u> 传递给它的子组件,但是子组件不能修改 <u>props</u> - <u>React.js 是单向数据流</u>,如果子组件需要修改父组件状态(数据),是通过回调函数方式来完成的。

props 函数

父组件在调用子组件的时候,传入一个函数类型的 <u>props</u>

<FriendGroup data={datas[key]} index={index} expandedIndex={expandedIndex} key={key} />

在子组件中根据具体的处理逻辑,调用父组件通过 <u>props</u> 传入的对应函数,类似 <u>on 事件</u> 的机制,同时子组件可以在调用回调函数的时候,传入一些数据

更新父组件

在父组件的代码中,通过 <u>setState</u> 重新渲染父组件,父组件的更新会重新渲染子组件

key

憋说话,看我!

class MyComponent extends React.Component {

    constructor(props) {
        super(props);
        
        this.state = {
            arr: ['html', 'css', 'javascript']
        }
        this.sort = this.sort.bind(this);
    }

    sort() {
        this.setState({
            arr: this.state.arr.sort( _ => Math.random() - .5 )
        });
    }

    render() {
        let {arr} = this.state;
        return(
            <div>
                <ul>
                    {arr.map( (v, k) => (
                        <li>
                            <input type="checkbox"/>
                            <span>{v}</span>
                        </li>
                    ) )}
                </ul>
                <button onClick={this.sort}>乱序</button>
            </div>
        );
    }
}

ReactDOM.render(
    <MyComponent />,
    document.getElementById('app')
);

通过控制台查看结构变化

渲染优化

  1. 用JS对象模拟DOM树 : <u>Virtual DOM</u>
  2. 通过 <u>DOM 操作</u> 把 <u>Virtual DOM</u> 生成真实的 <u>DOM</u>
  3. 后期的更新,会比较 新老 <u>Virtual DOM</u> ,通过 <u>diff</u> 算法优化更新操作

Virtual DOM

原生 <u>DOM</u> 对象结构太复杂,同时存在许多情况下用不到的一些特性

<u>Virtual DOM</u> 本质上与 原生 <u>DOM</u> 没有区别,都是 <u>JavaScript</u> 对象,只是它的结构更加简洁

<u>Diffing</u> 算法

<u>React.js</u> 不只是模拟了一个 <u>DOM</u>,同时在针对 <u>DOM</u> 进行渲染的时候

[图片上传失败...(image-ff9222-1613962628612)]

©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容