React 之 组件化开发

React 之 组件化开发_第1张图片
本文主讲解类组件,函数组件会在后续文章中学习

一、组件化开发

1. 概念

组件化是一种分而治之的思想:

  • 如果将一个页面中所有的处理逻辑放在一起,处理起来会变得非常复杂,不利于后续的管理以及扩展

  • 但如果讲一个页面拆分成一个个小的功能块,每个功能块完成属于自己这部分独立的功能,那么页面的管理和维护就变得非常容易了

2. react组件化

组件化是React的核心思想
React 之 组件化开发_第2张图片

3. 组件的应用

在开发中充分的利用组件化的思想 :

  • 尽可能的将页面拆分成一个个小的、可复用的组件

  • 让我们的代码更加方便组织和管理,并且扩展性也更强

4. 组件的分类

React的组件相对于Vue更加的灵活和多样,按照不同的方式可以分成很多类组件:
  • 根据组件的定义方式

  • 分成:函数组件(Functional Component )和类组件(Class Component) => 主要!

  • 根据组件内部是否有状态需要维护

  • 分成:无状态组件(Stateless Component )和有状态组件(Stateful Component)

  • 根据组件的不同职责

  • 分成:展示型组件(Presentational Component)和容器型组件(Container Component);

  • 还有很多组件的其他概念:比如异步组件、高阶组件...

这些概念有很多重叠,但是他们最主要是关注数据逻辑和UI展示的分离
  • 函数组件、无状态组件、展示型组件主要关注 => UI的展示

  • 类组件、有状态组件、容器型组件主要关注 => 数据逻辑

01 - 类组件

概念

类组件的定义有如下要求
  • 组件的名称是大写字符开头(无论类组件还是函数组件)

  • 类组件需要继承自 React.Component

  • 类组件必须实现render函数

在ES6之前,可以通过create-react-class 模块来定义类组件
但是目前官网建议我们使用ES6的class类定义

使用class定义一个组件:

  • constructor是可选的,我们通常在constructor中初始化一些数据

  • this.state中维护的就是我们组件内部的数据

  • render() 方法是 class 组件中唯一必须实现的方法

/**
 * 内部有单独导出Component
 * import React from 'react';
 * import { Component } from 'react';
 * 所以可以合并
 */

import React, { Component } from 'react';

export default class App extends Component {
  constructor() {
    super();
    this.state = {};
  }

  render() {
    return (
      

title

); } }

render函数的返回值

React 元素
通过jsx编写的代码就会被编译成React.createElement,所以返回的就是一个React元素
render() {
  return (
    

title

); }
数组或 fragments
react内部会一次遍历数组,展示在界面中
render() {
  // return [1, 2, 3, 4];
  return[
    

h1

, h2,

h3

,

h4

] }
字符串或数值类型
它们在 DOM 中会被渲染为文本节点
render() {
  // return 'afdsafdsa';
  return 123;
}
Portals
可以渲染子节点到不同的 DOM 子树中
布尔类型或 null

什么都不渲染

render() {
  // 界面上什么都没有 => undefined、null、boolean;
  
  // return undefined;
  // return null;
  // return true;
  return false;
}

02 - 函数组件

函数组件是使用function来进行定义的函数
只是这个函数会返回和类组件中render函数返回一样的内容

函数组件有自己的特点(当然,后面有hooks后,就不一样了):

  • 没有生命周期,也会被更新并挂载,但是没有生命周期函数

  • this关键字不能指向组件实例(因为没有组件实例)

  • 没有内部状态(state)

// 函数式组件 => 很单纯,只是为了展示数据
function App(props) {
  // 返回值: 和类组件中render函数返回的是一致
  return 

App Functional Component

} export default App

5. 题外话

Snippets 快捷键 - 插件

React 之 组件化开发_第3张图片
  • imr

  • import React from 'react'

  • rce

  • 生成类组件

  • rpce

  • 生成更优化的类组件

二、组件生命周期

1. 生命周期概念

很多的事物都有从创建到销毁的整个过程,这个过程称之为是生命周期

生命周期和生命周期函数的关系:

  • 生命周期是一个抽象的概念,在生命周期的整个过程,分成了很多个阶段

  • 比如装载阶段(Mount),组件第一次在DOM树中被渲染的过程

  • 比如更新过程(Update),组件状态发生变化,重新更新渲染的过程

  • 比如卸载过程(Unmount),组件从DOM树中被移除的过程;

React内部为了告诉我们当前处于哪些阶段,会对组件内部实现的某些函数进行回调
这些函数就是生命周期函数:
  • 比如实现componentDidMount函数:组件已经挂载到DOM上时,就会回调

  • 比如实现componentDidUpdate函数:组件已经发生了更新时,就会回调

  • 比如实现componentWillUnmount函数:组件即将被移除时,就会回调

  • 可以在这些回调函数中编写自己的逻辑代码,来完成自己的需求功能

ps : React生命周期时,主要谈的类的生命周期,因为函数式组件是没有生命周期函数的

2. 常用生命周期函数

图解析

React 之 组件化开发_第4张图片

constructor

如果不初始化 state 或不进行方法绑定,则不需要为 React 组件实现构造函数

constructor中通常只做两件事情

  • 通过给 this.state 赋值对象来初始化内部的state

  • 为事件绑定实例(this)

import { Component } from 'react';

export default class App extends Component {
  // 1. 第一执行
  constructor() {
    super();
    // 初始化内部的state
    this.state = {};
    // 给方法绑定实例(this)
    this.onClick = this.onClick.bind(this);
  }

  onClick() {
    console.log('click', this);
  }
  
  // 2. 第二执行
  render() {
    return 'ohohoh';
  }
}

componentDidMount

componentDidMount() 会在组件挂载后(插入 DOM 树中)立即调用

通常进行的操作 :

  • 依赖于DOM的操作可以在这里进行

  • 在此处发送网络请求就最好的地方(官方建议)

  • 可以在此处添加一些订阅(会在componentWillUnmount取消订阅)

import { Component } from 'react';

export default class App extends Component {
  // 1. 第一执行
  constructor() {
    super();
    this.state = {};
  }

  // 2. 第二执行
  render() {
    return 'ohohoh';
  }

  // 3. 第三执行 : 组件被渲染到dom,已挂载
  componentDidMount() {
    // 操作dom
    document.querySelector('#root');

    // 网络请求
    // axios.get().then().catch()

    // 添加订阅
  }
}

componentDidUpdate

componentDidUpdate() 会在更新后会被立即调用,首次渲染不会执行此方法

通常进行的操作 :

  • 当组件更新后,可以在此处对 DOM 进行操作

  • 如果你对更新前后的 props 进行了比较,也可以选择在此处进行网络请求

  • 例如,当 props 未发生变化时,则不会执行网络请求)

import { Component } from 'react';

export default class App extends Component {
  // 1. 第一执行
  constructor() {
    super();
    this.state = {
      message: '冲啊,宇智波海绵宝宝'
    };
    console.log('第一执行 : ', 'constructor');
  }

  // 2. 第二执行 || 第五执行
  render() {
    console.log('第二 || 第五执行 : ', 'render');

    const { message } = this.state;
    return (
      

{message}

); } // 4. 第四执行 onClick() { // 更改完成后,会重新执行render函数 this.setState({ message: '宇智波火炎阵!!!' }); console.log('第四执行 : ', 'onClick'); } // 3. 第三执行 componentDidMount() { console.log('第三执行 : ', 'componentDidMount'); } // 5. 第六执行 // 组件的Dom被更新完成 : Dom发生更新 componentDidUpdate() { console.log('第六执行 : ', 'componentDidUpdate'); } }

componentWillUnmount

componentWillUnmount() 会在组件卸载及销毁之前直接调用
import { Component } from 'react';

export default class App extends Component {
  render() {
    return (
      

我显示了

); } // 当App组件被移除时,会被调用该生命周期 componentWillUnmount() { console.log('第五执行 : ', 'componentWillUnmount'); } } // 其他页面的操作 { { /* 比如刚开始isShowCommp这个为true,后面变为false */ } isShowCommp && ; }

通常进行的操作 :

  • 在此方法中执行必要的清理操作

  • 清除 time

  • 取消网络请求

  • 清除在 componentDidMount() 中创建的订阅等

3. 不常用生命周期函数

React中还提供了一些过期的生命周期函数,这些函数已经不推荐使用 : React - 过往API

图解析

React 之 组件化开发_第5张图片

getDerivedStateFromProps

getDerivedStateFromProps() 在调用 render方法之前调用,在初始化和后续更新都会被调用
返回一个对象来更新 state, 如果返回 null 则不更新任何内容

参数 :

  • 第一个参数为即将更新的 props

  • 第二个参数为上一个状态的 state

  • 可以比较props 和 state来加一些限制条件,防止无用的state更新

getDerivedStateFromProps 是一个静态函数, 不能使用this , 也就是只能作一些无副作用的操作
说实话,不知道干嘛的,先留着

shouldComponentUpdate

shouldComponentUpdate() 在调用 render方法之前调用,控制是否重新执行render函数
该生命周期函数可以做性能优化,
import { Component } from 'react';

export default class App extends Component {
  render() {
    return (
      

我显示了

); } // 这么设定后,就算执行this.setState()后,render函数不会执行,页面不会刷新 shouldComponentUpdate() { return false; } }

getSnapshotBeforeUpdate

在React更新DOM 之前回调的一个函数
可以获取DOM更新前的一些信息(比如说滚动位置)
React 之 组件化开发_第6张图片

三、父子组件间的通信

1. 父组件传递子组件

  • 父组件通过 属性=值 的形式来传递给子组件数据

  • 子组件通过 props 参数获取父组件传递过来的数据

  • 也可以直接展开

React 之 组件化开发_第7张图片

代码 : code

父组件

import React, { Component } from 'react';

import Text from './Text';

export class App extends Component {
  render() {
    const arr = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
    return (
      
App {/** 传递属性 */}
); } } export default App;

子组件

import React, { Component } from 'react';

export class Text extends Component {

  // 如果这里不定义state,也可以不写contructor,内部会自动做这一步操作
  constructor(props) {
    // 相当于this.props = props
    super(props);
  }
  render() {
    // 直接使用
    const { name, arr } = this.props;
    console.log('props', name, arr);
    return (
      
{name}
    {arr.map(item => (
  • {item}
  • ))}
); } } export default Text;

参数验证 : propTypes

  • 可以使用Flow或者TypeScrip进行类型验证

  • 也可以通过 prop-types来进行参数验证

React 之 组件化开发_第8张图片
import React, { Component } from 'react';
// 1. 导入这个验证的包
import PropTypes from 'prop-types';

export class Text extends Component {
  constructor(props) {
    super(props);
    this.state = {
      name: 'start'
    };
  }
  render() {
    const { name, arr } = this.props;
    console.log('props', name, arr);
    return 
{name}
; } } // 2. 给类绑定一个propTypes Text.propTypes = { // 3. 设定name => 是个string类型,且是个必穿类型 name: PropTypes.string.isRequired, // 4. 设定arr => 是个array类型 arr: PropTypes.array }; export default Text;
更多的验证方式,可以参考官网: proptypes
  • 比如验证数组,并且数组中包含哪些元素

  • 比如验证对象,并且对象中包含哪些key以及value是什么类型

  • 比如某个原生是必须的,使用 requiredFunc: PropTypes.func.isRequired

参数默认值 : defaultProps

方式一

import React, { Component } from 'react';
import PropTypes from 'prop-types';

export class Text extends Component {
  constructor(props) {
    super(props);
    this.state = {
      name: 'start'
    };
  }
  render() {
    const { name, arr } = this.props;
    console.log('props', name, arr);
    return 
{name}
; } } Text.propTypes = { name: PropTypes.string, arr: PropTypes.array }; // 给参数设定默认值 Text.defaultProps = { name: '我是默认值', arr: ['我', '是', '默', '认', '值'] }; export default Text;

方式二

import React, { Component } from 'react';
import PropTypes from 'prop-types';

export class Text extends Component {
  // 给参数设定默认值
  static defaultProps = {
    name: '我是默认值',
    arr: ['我', '是', '默', '认', '值']
  };
  constructor(props) {
    super(props);
    this.state = {
      name: 'start'
    };
  }
  render() {
    const { name, arr } = this.props;
    console.log('props', name, arr);
    return 
{name}
; } } Text.propTypes = { name: PropTypes.string, arr: PropTypes.array }; export default Text;

2. 子组件传递父组件

通过props传递消息 : 父组件给子组件传递一个回调函数,子组件中调用这个函数即可

父组件

import React, { Component } from 'react';
import Text from './Text';

export class App extends Component {
  constructor() {
    super();
    this.state = {
      counter: 100
    };
  }

  // 2. 子组件回调该方法
  changeCount(count) {
    this.setState({
      counter: this.state.counter + count
    });
  }

  render() {
    const { counter } = this.state;
    return (
      
当前计数 : {counter} {/** 1. 传递方法到子组件 */} this.changeCount(count)} />
); } } export default App;

子组件

import React, { Component } from 'react';
import PropTypes from 'prop-types';

export class Text extends Component {
  btnClick(count) {
    console.log('count', count);
    // 2. 调用父组件传递过来的方法,并且传递参数过去
    this.props.addClick(count);
  }
  render() {
    return (
      
); } } // 1. 设定接受参数为方法,也可不设定 Text.propTypes = { addClick: PropTypes.func }; export default Text;

3. 案例

效果

React 之 组件化开发_第9张图片

父组件

import React, { Component } from 'react';
import Text from './Text';

export class App extends Component {
  constructor() {
    super();
    this.state = {
      list: ['首页', '详情', '个人'],
      currentIndex: 0
    };
  }
  itemChange(index) {
    this.setState({
      currentIndex: index
    });
  }

  render() {
    const { list, currentIndex } = this.state;
    return (
      
this.itemChange(index)} /> {list[currentIndex]}
); } } export default App;

子组件

css

.nav{
  display: flex;
  align-items: center;
  justify-content: center;
  list-style: none;
  margin: 0;
  padding: 0;
}
.nav .item{
  flex: 1;
  font-size: 25px;
  padding: 10px;
  text-align: center;
}
.nav .item.active{
  color:red;
  border-bottom: 1px solid red;
}

jsx

import React, { Component } from 'react';
import PropTypes from 'prop-types';
import './style.css';

export class Text extends Component {
  constructor(props) {
    super(props);

    this.state = {
      currentIndex: 0
    };
  }
  itemClick(index) {
    this.setState({
      currentIndex: index
    });
    this.props.itemChange(index);
  }
  render() {
    const { currentIndex } = this.state;
    const { list } = this.props;
    return (
      
    {list.map((item, index) => { return (
  • this.itemClick(index)}> {item}
  • ); })}
); } } Text.propTypes = { list: PropTypes.array.isRequired }; export default Text;

四、组件中的插槽 - slot

插槽 : 让使用者可以决定某一块区域到底存放什么内容

1. children实现插槽

每个组件都可以获取到 props.children:包含组件的开始标签和结束标签之间的内容
例如 : <子组件> 内容
  • 当内容只有一个元素时 : children === 内容

  • 当内容有多个元素时 : chindren 是一个数组

效果

React 之 组件化开发_第10张图片

父组件

import React, { Component } from 'react';
import NavBar from './NavBar';

export class App extends Component {
  render() {
    return (
      

h2

i

h3

b span
); } } export default App;

子组件

jsx

import React, { Component } from 'react';

import './index.css';

export class index extends Component {
  render() {
    // 当有多个子元素时,children是一个数组
    const { children } = this.props;
    return (
      
    {children.map((child, index) => { return (
  • {child}
  • ); })}
); } } export default index;

css

*{
  margin: 0;
  padding: 0;
}
.nav{
  display: flex;
  align-items: center;
  justify-content: center;
  list-style: none;
  margin: 0;
  padding: 0;
  height: 40px;
  margin-bottom: 10px;
  line-height: 40px;
}
.nav .item{
  flex: 1;
  height: 100%;
  font-size: 25px;
  text-align: center;
}
.nav .item:nth-child(1){
  background-color: #f00;
}
.nav .item:nth-child(2){
  background-color: #0f0;
}
.nav .item:nth-child(3){
  background-color: #00f;
}    

2. props实现插槽

通过children实现的方案虽然可行,但是有一个弊端
通过索引值获取传入的元素很容易出错,不能精准的获取传入的原生
React 之 组件化开发_第11张图片

五、祖孙组件通信 - Context

如果层级很多的话,一层层传递是非常麻烦,并且代码是非常冗余的
所以React提供了一个API:Context
  • Context 提供了一种在组件之间共享此类值的方式

  • 而不必显式地通过组件树的逐层传递 props

  • Context 设计目的是为了共享那些对于一个组件树而言是“全局”的数据

  • 例如当前认证的用户、主题或首选语言

React 之 组件化开发_第12张图片

相关API

React.createContext

创建一个需要共享的Context对象
  • 如果一个组件订阅了Context,那么这个组件会从离自身最近的那个匹配的 Provider 中读取到当前的context值

  • defaultValue是组件在顶层查找过程中没有找到对应的Provider,那么就使用默认值

// 封装成一个单独的js
import React from 'react';
export const MyContext = React.createContext();

Context.Provider

包裹需要使用数据的子组件
ps : 每个 Context 对象都会返回一个Provider React 组件,它允许消费组件订阅 context 的变化:
  • Provider 接收一个 value 属性,传递给消费组件

  • 一个 Provider 可以和多个消费组件有对应关系

  • 多个 Provider 也可以嵌套使用,里层的会覆盖外层的数据

  • 当 Provider 的 value 值发生变化时,它内部的所有消费组件都会重新渲染

import React, { Component } from 'react';
// 1. 导入
import { MyContext } from './context/MyContext';

import One from './One';

export class App extends Component {
  render() {
    const obj = { name: 'avc', age: 19 };
    return (
      
{/* 2. 包裹 => value:需要传递的数据*/} {/* 3. 子组件 */}
); } } export default App;

Class.contextType

在子组件中挂载一下context
ps : 挂载在 class 上的 contextType 属性会被重赋值为一个由 React.createContext() 创建的 Context 对象:
  • 可以使用 this.context 来消费最近 Context 上的那个值

  • 可以在任何生命周期中访问到它,包括 render 函数中

import React, { Component } from 'react';
import Two from './Two';

// 1. 导入相同的contenxt
import { MyContext } from './context/MyContext';

export class One extends Component {
  render() {
    // 3. 使用context
    console.log(this.context); // {name: 'avc', age: 19}
    return (
      
One {/* 子组件中的子组件,里面进行相同的操纵后也能拿到context传递的数据 */}
); } } // 2. 挂载过来 One.contextType = MyContext; export default One;

Context.Consumer

如果子组件是函数式组件的时候,使用Consumer
多个context的时候,也可以使用
// 1. 导入context
import { MyContext } from './context/MyContext';

function Two() {
  return (
    

Two

{/* 2. 使用MyContext.Consumer */} {/* 3. 这里使用一个函数,value就是传递过来的数据 */} {(value) => { return

{value.name}

; }}
); } export default Two;

多个Context使用

祖元素

import React, { Component } from 'react';
import { MyContext } from './context/MyContext';
import { MyContextTwo } from './context/MycontextTwo';

import One from './One';

export class App extends Component {
  render() {
    const obj = { name: 'aaa', age: 10 };
    return (
      
{/* 外层context*/} {/* 内层context*/} {/* 子组件 */}
); } } export default App;

子元素 : 类组件

import React, { Component } from 'react';

// 1. 导入相同的contenxt
import { MyContext } from './context/MyContext';
import { MyContextTwo } from './context/MycontextTwo';

export class One extends Component {
  render() {
    // 3. 使用挂载的context
    console.log(this.context); // {name: 'avc', age: 19}
    return (
      
{/* 4. 使用其他的context */} {/* {(value) =>
{value.name}
} */} {(value) => this.otherContext(value)}
); } // 可以提取出来 otherContext(value) { return
{value.name}
; } } // 2. 挂载其中一个context过来 One.contextType = MyContext; export default One;

子元素 : 函数组件

// 1. 导入context
import { MyContext } from './context/MyContext';
import { MyContextTwo } from './context/MycontextTwo';

function Two() {
  return (
    
{/* 其中一个context */} { (value) =>

{value.name}

}
{/* 另外一个context */} { (value) =>

{value.name}

}
); } export default Two;

默认值defaultValue

context

import React from 'react';
// 定义context的默认值
const defaultValue = { name: '我是默认值' };
export const MyContext = React.createContext(defaultValue);

祖元素

import React, { Component } from 'react';
// 1. 导入context
import { MyContext } from './context/MyContext';

import One from './One';

export class App extends Component {
  render() {
    const obj = { name: 'aaa', age: 10 };
    return (
      
{/* 2. 使用context.provider */} {/*3. 子组件没有在context的包裹中 */}
); } } export default App;

子元素

import React, { Component } from 'react';

// 1. 导入相同的contenxt
import { MyContext } from './context/MyContext';

export class One extends Component {
  render() {
    // 3. 这里拿到的就是默认数据
    console.log(this.context); // {name: '我是默认值'}

    return 
哈哈哈
; } } // 2. 挂载context过来 One.contextType = MyContext; export default One;

六、setState

1. 为什么使用setState

  • 开发中并不能直接通过修改state的值来让界面发生更新

  • React并没有实现类似于Vue2中的Object.defineProperty或者Vue3中的Proxy的方式来监听数据的变化

  • 必须通过setState来告知React数据已经发生了变化

  • 在组件中并没有实现setState的方法,为什么可以调用

  • 原因很简单,setState方法是从Component中继承过来的

2. setState用法

用法一

原理 : Object.assign(this.state,newState)
ps : 把新传入的对象和原来的对象合并,若有相同的,取代
this.setState({
   name: 'bbb',
   age: 18
});

用法二

/**
 * 传入回调函数 => 返回一个新对象
 *    好处一 : 可以进行逻辑编写
 *    好处二 : 当前回调函数会将组件之前的state和组件接收的props传递进来
 */
this.setState((preState, props) => {
  // 返回一个新对象
  return {
    name: 'bbb',
    age: 18
  };
});

3. setState异步更新

setState的更新是异步的
this.setState({
  name: 'bbb',
  age: 18
});
console.log(this.state.name, this.state.age); // aaa 10
  • 最终打印结果是aaa 10

  • 可见setState是异步的操作,并不能在执行完setState之后立马拿到最新的state的结果

01 - 优点

  • setState设计为异步,可以显著的提升性能

  • 如果每次调用 setState都进行一次更新,那么意味着render函数会被频繁调用,界面重新渲染,这样效率是很低

  • 最好的办法应该是获取到多个更新,之后进行批量更新

  • 如果是同步,更新了state,但是还没有执行render函数,那么state和props不能保持同步

  • state和子组件的props不能保持一致性,会在开发中产生很多的问题

02 - 获取异步的结果

回调函数

setState接受两个参数:第二个参数是一个回调函数,这个回调函数会在更新后会执行
itemClick() {
  this.setState(
    {
      name: 'bbb',
      age: 18
    },
    // 使用回调函数
    () => {
      console.log(this.state.name, this.state.age); // bbb 18
    }
  );
}

生命周期

itemClick() {
  this.setState({
    name: 'bbb',
    age: 18
  });
}
componentDidUpdate() {
  // 生命周期获取修改后的值
  console.log(this.state.name, this.state.age); // bbb 18
}

03 - React版本

React18之前

分成两种情况:

  • 在组件生命周期或React合成事件中,setState是异步

  • 在setTimeout或者原生dom事件中,setState是同步

React 之 组件化开发_第13张图片

React18之后

在React18之后,默认所有的操作都被放到了批处理中(异步处理)
React 之 组件化开发_第14张图片
默认是异步,可以通过 flushSync 设置成同步
import React, { Component } from 'react';
// 1. 导入
import { flushSync } from 'react-dom';

export class App extends Component {
  constructor() {
    super();
    this.state = {
      name: 'aaa',
      age: 10
    };
  }
  itemClick() {
    // 2. 包裹
    flushSync(() => {
      this.setState({
        name: 'bbb'
      });
    });
    // 3. 获取到值 => 会先执行rander,然后再执行这里
    console.log(this.state); // {name:'bbb',age:10}
  }

七、React性能优化

1. Diff算法优化

概念

渲染流程

React 之 组件化开发_第15张图片
  • rander函数中返回JSX

  • JSX会创建对应的ReactElement

  • ReactElement最终会形成一个树结构,这个树结构就是虚拟DOM

  • 最后React会根据虚拟DOM渲染成真实DOM

更新流程

React 之 组件化开发_第16张图片

diff算法

  • React在props或state发生改变时,会调用React的render方法,会创建一颗不同的树

  • React需要基于这两颗不同的树之间的差别来判断如何有效的更新UI

  • 这个判断的过程就是diff算法

优化

React对这个算法进行了优化,将其优化成了O(n)

同层节点之间相互比较,不会垮节点比较

React 之 组件化开发_第17张图片

不同类型的节点,产生不同的树结构

React 之 组件化开发_第18张图片

开发中,可以通过key来指定哪些节点在不同的渲染下保持稳定

React 之 组件化开发_第19张图片

2. 类组件性能优化SCU

问题

只要是修改了 App中的数据,所有的组件都需要重新render,进行diff算法,性能必然是很低的

事实上,很多的组件没有必须要重新render
调用render应该有一个前提,就是依赖的数据(state、 props)发生改变时,再调用自己的render方法
import React, { Component } from 'react';
import Home from './Home';

export class App extends Component {
  constructor() {
    super();
    this.state = {
      name: 'aaa',
      age: 10
    };
  }
  itemClick() {
    this.setState({
      name: 'bbb'
    });
  }

  /**
   * 问题一 : 调用this.setState后,就算没有相关依赖的变量,
              Home子组件乃至子组件的子组件,它们的render方法也会执行

   * 问题二 : this.setState({
                name: 'aaa'
              })  没有修改内容,同样会全部重新渲染

   * 效率极其低下
   */

  render() {
    console.log('render执行');
    const { name, age } = this.state;
    return (
      

this.itemClick()}> {name} - {age}

      {/* 可以不必重新渲染 */}
); } } export default App;

方法

通过shouldComponentUpdate方法来控制render方法是否被调用

shouldComponentUpdate

React 之 组件化开发_第20张图片
import React, { Component } from 'react';
import Home from './Home';

export class App extends Component {
  constructor() {
    super();
    this.state = {
      name: 'aaa',
      age: 10
    };
  }
  itemClick() {
    this.setState({
      name: 'bbb'
    });
  }

  shouldComponentUpdate(nextProps, nextState) {
    console.log(nextProps, '没有props就为空');
    console.log(`旧 : ${this.state.name}, 新 : ${nextState.name}`); // aaa  bbb

    // 当不同时,才重新渲染, render函数执行
    // return this.state.name !== nextState.name || this.state.age !== nextState.age;

    // 如果作为子组件,有props的情况下,还需要队props进行对比
    return (
      this.state.name !== nextState.name ||
      this.state.age !== nextState.age ||
      // props
      this.props.message !== nextProps.message
    );
  }

  render() {
    console.log('render执行');
    const { name, age } = this.state;
    return (
      

this.itemClick()}> {name} - {age}

{/* 可以不必重新渲染 */}
); } } export default App;

PureComponent

如果所有的类,都需要手动来实现 shouldComponentUpdate,那么会给开发增加非常多的工作量
React已经考虑到了这一点,所以React已经默认实现好了
将class继承自PureComponent
// 1. 不导入Component,导入PureComponent
import React, { PureComponent } from 'react';
import Home from './Home';

// 2. 不继承Component,继承PureComponent
export class App extends PureComponent {

  render() {
    console.log('render执行');
    return (
      

App

); } } export default App;
PureComponent : 本质是进行了一个浅层比较 => 只比较第一层,不比较深层
React 之 组件化开发_第21张图片

3. 函数组件性能优化SCU

需要使用一个高阶组件memo => 包裹一下函数组件即可
import { memo } from 'react';

// 未使用memo的函数组件
// export default function Home(props) {
//   console.log('home render');
//   return 

123:{props.name}

; // } // 使用了memo的函数组件 export default memo(function Home(props) { console.log('home render'); return

123:{props.name}

; });

4. 不可变数据的力量

不要直接操作,而是赋予新的地址
ps : 在setState中,进行结构即可,不管是整体,还是修改单个属性依旧如此
import React, { PureComponent } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      friend: ['a', 'b', 'c']
    };
  }
  addClick() {
    const newEl = 'd';
    /**
     * 直接修改原有state,重新设置一遍
       在PureComponent中,是不会重新渲染页面的
       因为shouldComponentUpdate回调函数里,nextState - this.state两个值是一样的,就不会触发页面的刷新
     */
    // this.state.friend.push(newEl);
    // this.setState({
    //   friend: this.state.friend
    // });

    /**
     * 正确方式一
     */
    // this.state.friend.push(newEl);
    // this.setState({
    //   // 相当于拿新的数组,覆盖了原来的数组,地址不一样了
    //   friend: [...this.state.friend]
    // });
    /**
     * 正确方式二
     */
    const friend = [...this.state.friend];
    friend.push(newEl);
    this.setState({
      friend
    });
  }

  render() {
    console.log('render执行');
    const { friend } = this.state;
    return (
      
    {friend.map((item, index) => { return
  • {item}
  • ; })}
); } } export default App;

八、Ref获取DOM和组件

Ref获取DOM

方式一 : 传入字符串

使用时通过 this.refs.传入的字符串格式 获取对应的元素 => 已经被废弃了
import React, { PureComponent } from 'react';

export class App extends PureComponent {
  onNativeClick() {
    // 2. 通过ref属性获取到DOM节点 不过已经被废弃了
    console.log('this.refs.coderRef', this.refs.coderRef);
  }
  render() {
    return (
      
{/* 1. 增加一个ref属性 */}

this.onNativeClick()}> hello world

); } } export default App;

方式二 : 传入一个对象 => 推荐

对象是通过 React.createRef() 方式创建出来的
使用时获取到创建的对象其中有一个current属性就是对应的元素
// 通过createRef获取dom元素
import React, { PureComponent, createRef } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {};
    // 1. 通过createRef创建一个ref对象,因为不需要响应式,所以不需要放在state中
    this.coderRef = createRef();
  }
  onNativeClick() {
    // 3. 通过current属性获取dom元素
    console.log('this.coderRef', this.coderRef.current);
  }
  render() {
    return (
      
{/* 2. 绑定到ref属性上 */}

this.onNativeClick()}> hello world

); } } export default App;

方式三 : 传入一个函数

该函数会在DOM被挂载时进行回调,这个函数会传入一个 元素对象,我们可以自己保存
使用时,直接拿到之前保存的元素对象即可
import React, { PureComponent } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {};
    // 1. 创建一个ref对象
    this.coderRef = null;
  }
  onNativeClick() {
    // 3. 拿到dom元素
    console.log('this.coderRef', this.coderRef);
  }
  render() {
    return (
      

(this.coderRef = el)} onClick={() => this.onNativeClick()}> hello world

); } } export default App;

Ref获取类组件

ref 的值根据节点的类型而有所不同:
  • 当 ref 属性用于 HTML 元素时

  • 构造函数中使用 React.createRef() 创建的 ref 接收底层 DOM 元素作为其 current 属性

  • 当 ref 属性用于自定义 class 组件时

  • ref 对象接收组件的挂载实例作为其 current 属性

import React, { PureComponent, createRef } from 'react';

// 子组件
class Text extends PureComponent {
  text() {
    console.log('Text组件的text方法');
  }
  render() {
    return 

Text

; } } // 父组件 export class App extends PureComponent { constructor() { super(); this.state = {}; // 1. 通过createRef创建一个ref对象 this.textRef = createRef(); } onComponentClick() { // 3. 拿到子组件实例 console.log('this.textRef', this.textRef.current); // 4. 调用子组件的方法 this.textRef.current.text(); } render() { return (
{/* 2. 绑定到ref属性上*/}
); } } export default App;

Ref获取函数组件

  • 不能在函数组件上使用 ref 属性,因为函数组件没有实例

  • 但是某些时候,可能想要获取函数式组件中的某个DOM元素

  • 这个时候可以通过 React.forwardRef 高阶组件

import React, { PureComponent, createRef, forwardRef } from 'react';

// 子组件
/**
 *  forwardRef 用于转发 ref,使得 ref 可以在函数组件中传递
 * 1. forwardRef 接受一个渲染函数,函数的参数为 props 和 ref
 * 2. forwardRef 返回一个组件,这个组件接受 props 和 ref 两个参数
 */
const Text = forwardRef(function (props, ref) {
  return (
    

hello - 001

{/* 3. 把传递过来的ref绑定到这个元素上 */}

hello - 002

hello - 003

); }); // 父组件 export class App extends PureComponent { constructor() { super(); this.state = {}; // 1. 通过createRef创建一个ref对象 this.textRef = createRef(); } onComponentClick() { // 4. 拿到子组件的某一个dom元素 console.log('this.textRef', this.textRef.current); } render() { return (
{/* 2. 绑定到ref属性上*/}
); } } export default App;

九、受控组件 && 非受控组件

1. 受控组件

一般来说,当给表单组件绑定了value属性后,该表单组件就变成了受控组件
只能通过React来使用 setState()进行更新
  • 在表单元素上设置了 value 属性,因此显示的值将始终为 this.state.value

  • 这使得 React 的 state 成为唯一数据源

  • handleUsernameChange 在每次按键时都会执行并更新 React 的 state

  • 因此显示的值将随着用户输入而更新

  • 被 React 以这种方式控制取值的表单输入元素就叫做“受控组件

React 之 组件化开发_第22张图片

input

import React, { PureComponent } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      // 1. 定义需要绑定到输入框的数据
      username: '',
      password: ''
    };
  }
  // 3. 监听到值改变后,更新message的值,再修改输入框的值
  inputHandleChange(e) {
    this.setState({
      // 使用计算属性
      [e.target.name]: e.target.value
    });
  }
  submitHandle() {
    console.log(this.state);
  }
  render() {
    const { username, password } = this.state;
    return (
      
{/* 2. 绑定到value属性上 => 监听onChange方法(不监听的话,无法输入值) */}
); } } export default App;

checkbox

import React, { PureComponent } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      // 1. 定义需要绑定到输入框的数据
      agree: false,
      hobbies: [
        { key: 'sing', value: '唱', isChecked: false },
        { key: 'dance', value: '跳', isChecked: false },
        { key: 'rap', value: 'rap', isChecked: false }
      ]
    };
  }
  // 3. 监听到值改变后,更新message的值,再修改输入框的值
  checkHandleChange(e, index) {
    // 单选
    if (index === -1) {
      return this.setState({
        [e.target.name]: e.target.checked
      });
    }
    // 多选
    const { hobbies } = this.state;
    hobbies[index].isChecked = e.target.checked;
    this.setState({
      // 使用计算属性
      hobbies: [...hobbies]
    });
  }
  submitHandle() {
    console.log(this.state);
    console.log(
      '爱好',
      this.state.hobbies.filter((item) => item.isChecked).map((item) => item.key)
    );
  }
  render() {
    const { agree, hobbies } = this.state;
    return (
      
{/* 2. 绑定到value属性上 => 监听onChange方法(不监听的话,无法输入值) */}
单选 ===
多选 === {hobbies.map((item, index) => { return ( ); })}
); } } export default App;

select

import React, { PureComponent } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      fruits: 'banana',
      fruitsArr: ['banana']
    };
  }
  // 3. 监听到值改变后,更新message的值,再修改输入框的值
  selectHandleChange(e, type) {
    // 单选
    if (type === 'single') {
      return this.setState({
        fruits: e.target.value
      });
    }

    // 多选  => e.target.selectedOptions 可以拿到所有多选的数据,按住shift多选
    const options = Array.from(e.target.selectedOptions);
    const optionsValue = options.map((item) => item.value);
    this.setState({
      // 使用计算属性
      fruitsArr: [...optionsValue]
    });
  }
  submitHandle() {
    console.log(this.state);
  }
  render() {
    const { fruits, fruitsArr } = this.state;
    return (
      
{/* 2. 绑定到value属性上 => 监听onChange方法(不监听的话,无法输入值) */}
单选 ===



多选 ===
); } } export default App;

2. 非受控组件

React推荐大多数情况下使用 受控组件 来处理表单数据
  • 一个受控组件中,表单数据是由 React 组件来管理的

  • 另一种替代方案是使用非受控组件,这时表单数据将交由 DOM 节点来处理

  • 如果要使用非受控组件中的数据,可以使用 ref 来从DOM节点中获取表单数据

代码

import React, { PureComponent, createRef } from 'react';

export class App extends PureComponent {
  constructor() {
    super();
    this.state = {
      message: '123'
    };
    // 1. 初始化ref
    this.inputRef = createRef();
  }

  componentDidMount() {
    // 3. 可以通过ref获取到DOM元素,然后绑定事件,监听值的变化
    this.inputRef.current.addEventListener('input', (e) => {
      console.log(e.target.value);
    });
  }

  submitHandle() {
    // 4. 通过ref获取到值
    console.log(this.inputRef.current.value);
  }
  render() {
    const { message } = this.state;
    return (
      
{/* 2. 绑定ref,并使用defaultValue */}
); } } export default App;

效果

React 之 组件化开发_第23张图片

十、高阶组件

1. 定义

高阶组件是参数为组件,返回值为新组件的函数
  • 高阶组件的英文是 Higher-Order Components,简称为 HOC

  • 高阶组件本身不 是一个组件,而是一个函数

  • 这个函数的参数是一个组件,返回值也是一个组件


/**
 * 高阶组件是一个函数,参数是一个组件,返回值也是一个组件
 *
 * 高阶组件的作用是为了增强组件,给组件添加一些额外的功能
 * 比如:给组件添加一些公共的状态、公共的逻辑、公共的生命周期
 * 或者是给组件添加一些公共的UI
 * 或者是给组件添加一些公共的数据
 * 或者是给组件添加一些公共的方法等等
 * 总之,高阶组件的作用就是为了增强组件,给组件添加一些额外的功能。
 */
// 定义高阶组件,参数是一个组件,返回值也是一个组件,
function hoc(wrapperComponent) {
  // 1. 定义类组件
  return class extends PureComponent {};
  
  // 2. 定义函数组件
  // return function () {
  //   return 
函数组件
; // } }
高阶组件并不是React API的一部分,它是基于React的组合特性而形成的设计模式

2. 应用

props的增强

不修改原有代码的情况下,添加新的props

代码

import React, { PureComponent } from 'react';

// 1. 定义高阶组件
function enhanceProps(OriginComponent) {
  return class extends PureComponent {
    constructor(porps) {
      super(porps);
      this.state = { name: 'coder', age: 18 };
    }
    render() {
      // 2. 注入props 本身的props和传入的props,都要注入到OriginComponent中
      return ;
    }
  };
}

// 3. 函数组件使用高阶组件
const Home = enhanceProps(function (props) {
  const str = 'home';
  return (
    

Home

父组件传递过来的props ={'>'} level : {props.level}

增强的props ={'>'} name : {props.name} - age : {props.age}

本身定义的数据 ={'>'} str : {str}

); }); // 4. 类组件使用高阶组件 const Text = enhanceProps( class extends PureComponent { constructor(props) { super(props); this.state = { str: 'text' }; } render() { const { name, age, level } = this.props; const { str } = this.state; return (

Text:

父组件传递过来的props ={'>'} level : {level}{' '}

增强的props ={'>'} name : {name} - age : {age}

本身定义的数据 ={'>'} str : {str}{' '}

); } } ); export class App extends PureComponent { render() { return (
); } } export default App;

效果

React 之 组件化开发_第24张图片

Context共享

共享全局主题颜色

context

import React from 'react';

export const ThemeContext = React.createContext();

父组件

import React, { PureComponent } from 'react';
import { ThemeContext } from './context/ThemeContext';
import Home from './components/Home';

export class App extends PureComponent {
  render() {
    return (
      
App {/* 使用context提供主题数据 */}
); } } export default App;

高阶组件

// 使用context
import { ThemeContext } from '../context/ThemeContext';

export default function withTheme(OriginComponent) {
  // 返回一个函数组件
  return (props) => {
    // 渲染的内容
    return (
      
        {/* 使用context共享数据 */}
        {
          (value) => {
            return ;
          }
        }
      
    );
  };
}

子组件

import React, { PureComponent } from 'react';
import withTheme from '../hoc/with-theme';

export class Home extends PureComponent {
  render() {
    return (
      
Home {/* 使用高阶组件中context的数据 */} {this.props.color} {/* red */} {this.props.size} {/* 30 */}
); } } // 实际导出的是一个高阶组件 export default withTheme(Home);

渲染判断鉴权

父组件

import React, { PureComponent } from 'react';
import Home from './components/Home';

export class App extends PureComponent {
  loginClick() {
    // 模拟登录
    sessionStorage.setItem('token', 123321);
    
    // 因为没有调用setState,不会重新渲染页面
    // 所以进行强制刷新, 重新渲染组件, 但是不推荐使用
    this.forceUpdate();
  }
  render() {
    return (
      
App
); } } export default App;

高阶组件

export default function loginAuth(OriginComponent) {
  return (props) => {
    // 1. 从localStorage中获取token
    const token = sessionStorage.getItem('token');
    // 2. 判断token是否存在
    if (token) {
      // 2.1 如果存在,渲染组件
      return ;
    } else {
      // 2.2 如果不存在,跳转到登录页面
      return 

请先登录

; } }; }

子组件

import React, { PureComponent } from 'react';
import loginAuth from '../hoc/login-auth';

export class Home extends PureComponent {
  render() {
    return 
Home
; } } // 实际导出的是一个高阶组件 export default loginAuth(Home);

3. 意义

利用高阶组件可以针对某些React代码进行更加优雅的处理

01 - mixin

早期的React有提供组件之间的一种复用方式是mixin,目前已经不再建议使用:
  • Mixin 可能会相互依赖,相互耦合,不利于代码维护

  • 不同的Mixin中的方法可能会相互冲突

  • Mixin非常多时,组件处理起来会比较麻烦,甚至还要为其做相关处理,这样会给代码造成滚雪球式的复杂性

02 - hoc

HOC也有自己的一些缺陷:
  • HOC需要在原组件上进行包裹或者嵌套,如果大量使用HOC,将会产生非常多的嵌套

  • 这让调试变得非常困难

  • HOC可以劫持props,在不遵守约定的情况下也可能造成冲突

03 - hooks

Hooks的出现,是开创性的,它解决了很多React之前的存在的问题
  • 比如this指向问题

  • 比如hoc的嵌套复杂度问题

十一、Portals

某些情况下,我们希望渲染的内容独立于父组件,甚至是独立于当前挂载到的DOM元素中
默认都是挂载到id为root的DOM元素上的

Portal 提供了一种将子节点渲染到存在于父组件以外的 DOM 节点的优秀的方案:

  • 第一个参数(child)是任何可渲染的 React 子元素

  • 例如一个元素,字符串或 fragment

  • 第二个参数(container)是一个 DOM 元素

1. 基本使用

在根组件新创建一个

代码

import React, { PureComponent } from 'react';
// 1. 导入 createPortal,用于创建一个传送门
import { createPortal } from 'react-dom';

export class App extends PureComponent {
  render() {
    return (
      
App {/* 2. 使其挂载到根组件之外的coder -> div中 */} {createPortal(

我是传送门

, document.querySelector('#coder'))}
); } } export default App;

效果

React 之 组件化开发_第25张图片

2. 封装

根文件index.html

React 之 组件化开发_第26张图片

父组件

import React, { PureComponent } from 'react';
import Coder from './components/Coder';

export class App extends PureComponent {
  render() {
    return (
      
App {/* 在子组件中间写内容,类似插槽的效果 */}

coder

hello world
); } } export default App;

子组件

import { PureComponent } from 'react';
import { createPortal } from 'react-dom';

export class Coder extends PureComponent {
  render() {
    // createPortal(要渲染的内容, 挂载的DOM节点)
    // this.props.children => 是从父组件传递过来的内容
    return createPortal(this.props.children, document.getElementById('coder'));
  }
}

export default Coder;

效果

React 之 组件化开发_第27张图片

十二、fragment

之前的开发中,总是在一个组件中返回内容时包裹一个div元素
如果希望可以不渲染这样一个div应该如何操作
  • 使用Fragment

  • Fragment 允许你将子列表分组,而无需向 DOM 添加额外节点

  • 类似vue的template、小程序的block

1. 基本使用

import React, { PureComponent, Fragment } from 'react';

export class App extends PureComponent {
  render() {
    return (
      // 把Fragment当做一个占位符使用,不会渲染到页面上
      
        

App标题

冲啊

); } } export default App;

2. 语法糖

Fragment的短语法:
  • 它看起来像空标签 <>

  • 但是,如果需要在Fragment中添加key,那么就不能使用短语法

render() {
  return (
    // Fragment语法糖
    <>
      

App标题

冲啊

);】 }

十三、StrictMode

StrictMode 是一个用来突出显示应用程序中潜在问题的工具:
  • 与 Fragment 一样,StrictMode 不会渲染任何可见的 UI

  • 它为其后代元素触发额外的检查和警告

  • 严格模式检查仅在开发模式下运行

  • 它们不会影响生产构建

可以为应用程序的任何部分启用严格模式
不会对 Header 和 Footer 组件运行严格模式检查

1. 全局启动

import ReactDOM from 'react-dom/client';

import App from './App.jsx';
import { StrictMode } from 'react';

const root = ReactDOM.createRoot(document.querySelector('#root'));

// 开启严格模式,检查不安全的生命周期,过时的ref等
root.render(
  
    
  
);

2. 部分启动

import React, { PureComponent, StrictMode } from 'react';
import Home from './components/Home';
import Coder from './components/Coder';

export class App extends PureComponent {
  render() {
    return (
      <>
        

App

{/* 给Home组件开启严格模式 => Home以及它的所有后代元素都将进行检查 */} ); } } export default App;

3. 严格模式检查的内容

识别不安全的生命周期

React 之 组件化开发_第28张图片

使用过时的ref API

React 之 组件化开发_第29张图片

检查意外的副作用

constructor、render、以及很多的生命周期函数,会在第一次渲染的时候, 刻意执行两次
  • 这个组件的constructor会被调用两次

  • 这是严格模式下故意进行的操作,让你来查看在这里写的一些逻辑代码被调用多次时,是否会产生一些副作用

  • 在生产环境中,是不会被调用两次的

使用废弃的findDOMNode方法

在之前的React API中,可以通过findDOMNode来获取DOM,已经不推荐使用了

检测过时的context API

  • 早期的Context是通过static属性声明Context对象属性,通过getChildContext返回Context对象等方式来使用Context的

  • 目前这种方式已经不推荐使用

你可能感兴趣的:(React,react,reactjs,前端,前端框架,组件化开发)