React

Jesse

React

1. What is React

  • CORE: Design by Component; 组合 & 嵌套 来构成整体页面
  • 数据流单向响应, 更安全更快

2. State & Props

  • state & setState 在 constructor 中初始化, 在更新内容后会重新调用render. 组件内部管理
  • props 是组件的 input parameters, from its parent class, 由外部传入
  • props 不可修改. Re-rendering child component by sending new props from outside

3. super() & super(props)

  • super 代替的是 parent constructor:
    • super(name) == sup.prototype.constructor.call(this.name)
  • child does not have this until inherit from parent by super(props)
  • if we don’t need this.props, then no need for super(props)
  • if super(), then this.props = undefined

4. Components

4.1 Class Component

State & lifecycle methods manageable but not friendly for new code (render components in JSX style)

1
2
3
4
5
6
7
8
9
10
11
12
class component extends React.Component {
constructur(props){
super(props);
...
}

// Other Life cycle methods
componentDidMount() {}
...

render () { return (); } // return in jsx style
}

4.2 Function Component

  • 创建 Component 可以通过 React.createElement(..); 被 babel 转化成 React.createClass
  • 适合无状态 / use Hooks
  • Manage State & lifecycle methods by hooks (return react component)
1
2
3
function MyComponent(props: Props) {
return <div>Hello from a function component!</div>;
}

4.3 区别 - state

Function Component 需要 useState 管理状态

4.4 区别 - Lifecycle

Function Component 不存在生命周期, 因为 hook 都继承于R.C, 但可以用useEffect代替生命周期的作用 -> componentDidMount & componentWillUnmount

4.5 区别 - 调用方式

1
2
const res = functionComponent(props);
const res = new classComponent(props);

4.6 区别 - 获取渲染值

Class Component 的this.props is constant unless this changes
Function Component does not have this, then props is constant

5. 受控组件 & 非受控组件

  • 受控组件 - 组件状态相应外部数据: onChange. 比如 input 表单数据
  • 非受控组件 - 在初始化时接受外部数据, 然后内部存储: React.createRef(). 比如 form 输入

6. React Event - 合成 SyntheticEvent

  • React 模拟原生 DOM 事件所有能力的事件对象: 事件注册, 合成, 冒泡, 派发.. (浏览器原生事件的跨浏览器包装器)
  • e.nativeEvent: 原生 DOM 事件
    1
    <Button onClick={(e) => console.log(e.nativeEvent)} />

6.1 事件监听器

  • onClick没有被作为 delegate 绑定到节点上, 而是把所有事件绑定到结构最外层, 让统一的事件进行监听
  • 维持一个映射来保存组件内部的事件监听 & 处理函数. update when component mount / unmount
  • 简化了事件处理和回收机制

6.2 合成事件和原生事件的处理顺序

  1. 真实 DOM 元素触发后: 先 document 对象 (原生事件componentDidMount), 再 React 事件 (onClick)
  2. 执行 document 上挂载的事件

6.3 冒泡

  • 阻止: e.stopPropagation() - 无法阻止 React 本身的冒泡机制
  • 阻止和最外层 document 上事件的冒泡: e.nativaEvent.stopImmediatePropagation()
  • 阻止和最外层 document 上事件的冒泡: 判断e.target避免

7. 事件绑定 - this

bind 会改变 this 指向, But avoid re-bind when re-rendering

1
2
3
4
5
6
7
// slow
<div onClick={this.handleClick.bind(this)}>
<div onClick={e => this.handleClick(e)} />

// fast:
this.handleClick = this.handleClick.bind(this) // in constructor
<div onClick={this.handleClick} />

8. CSS usage

  • 局部 css - 防止污染

8.1 组件内 - style

<div style = {{backgroundColor: "red"}} />

  • 无冲突, 写太多了容易混乱

8.2 import .css

import './App.css';

  • 全局生效, 样式之间会互相影响

8.3 import .module.css

import './App.module.css';

  • 引入的 css 不会影响后代组件
  • 需要配置modules:true in webpack配置文件
  • 不方便动态修改样式, 需要用内联方式编写

8.4 CSS in JS - 由 JS 生成的的 CSS

  • 第三方库: MUI

9. React lifecycle 不同阶段

9.1 创建

  • constructor
    • get parent props
    • 初始化 state, 挂载方法到 this, …
  • getDerivedStateFromProps
    • static
    • 比较props & state添加限制条件, 防止无用 state 更新
    • 返回新的对象作为 state 或者 null 表示不需要更新
  • render
    • 渲染 DOM
    • 不能用setState
  • componentDidMount
    • 挂载到真实 DOM 后执行
    • 多用于执行数据获取, 事件监听…

9.2 更新

  • getDerivedStateFromProps
  • shouldComponentUpdate
    • boolean: 通过比较props & state
    • 不建议深层比较, 影响效率
    • 不能调用setState - loop
  • render
  • getSnapshotBeforeUpdate
    • DOM 还没更新
    • 获取组件更新前的信息: 组件的滚动位置…
    • 返回的Snapshot value 被作为参数传入componentDidUpdate
  • componentDidUpdate
    • 更新结束后
    • 比较props & state变化做对应操作 - 修改数据

9.3 卸载

componentWillUnmount

  • 卸载前, 清理监听事件, 取消订阅网络请求…
  • 无法被再次挂载, 只能重新创建

10 组件通信

单向数据流

10.1 Parent -> Child

1
<EmailInput email="123@gmail.com" />

10.2 Child -> Parent

使用函数回调, 可以用onSet也可以bind

1
2
3
4
5
6
7
8
// parent
<Child onSetData = {setData} />
<Child getData = {this.setState.bind(this)} />

// child
<button onClick = {onSetData(100)} />
<button onClick = {this.getData.bind(this, 100)} />

10.3 Between Siblings

父组件作为中间层

1
2
<SiblingA count = {this.state.count} />
<SiblingB count = {this.state.count} />

10.4 Parent -> descendant

  • React.createContext - 看 Hook 部分

10.5 Between un-connected components

使用 redux 进行全局资源管理

11. 看懂什么是 this

  • def: Scope / Context of its use

11.1 Global scope

1
2
3
// this = Window
// this = {frames: Window, postMessage: ƒ, blur: ƒ, focus: ƒ, …}
this.alert("test") = Window.alert("test");

11.2 Function scope

this 取决于 function 如何被 call

  • explicitly set:

    • call, apply, bind the value of this in a function (include Arrow function)
    • object is a local variable (block scoped)
      • 处于 global scope 时, this 和 object 无关
      • 处于 class 时, this 和 object 有关
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    const object = { a: 5, b: 7 };
    const f = function (c, d) {
    return this.a + this.b + c + d;
    };

    // call
    f.call(object, 12, 4); // 5 + 7 + 12 + 4

    // apply
    f.apply(object, [12, 4]); // 5 + 7 + 12 + 4

    // bind
    const bindF = f.bind(object);
    bindF(12, 4); // 5 + 7 + 12 + 4
    • Arrow function
      • 无法修改 this 绑定
      • 不可位于 global 位置
        • 因为箭头函数不会创建自己的 this,它的 this 取决于定义时的词法作用域。
        • 在这种顶层声明的场景里, this = undefined (strict) 或者是 global scope, 但绝对不会自动指向定义的 object.
        • 直接使用即可,而不是写 this.object
1
2
3
4
5
6
7
8
const f = () => {
return this.object.a + this.object.b
}
// equivalent
constructor() { // 本质是把f放到constructor中
this.f = () => { /* ... */ };
...
}
  • not set:
    • this defaults to global context
    • strict mode: this = it’s explicit value
      • if not set, this = undefined
        1
        2
        3
        4
        5
        6
        7
        const f = function () {
        return this;
        }; // Window
        const f = function () {
        "use strict";
        return this;
        }; // undefined

11.3 Object context

this refer to the object to itself.

1
2
3
4
5
6
7
8
const object = {
a: 2,
b: 3,
f: function () {
return this.a + this.b;
},
};
object.f(); // returns 2 + 3

底层逻辑

JSX

总结:

  • 用于描述 component 内容
  • 是 JS 的语法扩展, 类似模版语言但是充分具备 JS 的能力
    • 被编译为React Element : createElement(type, config, children)
    • 因为是扩展: 需要用 Babel 使 JSX 支持浏览器
  • Babel
    • 工具链, 将 ECMAScript 代码转换为向后兼容的 JS 语法来防止不兼容
  • createElement 参数中介 - 数据格式化 => ReactElement:
    • ![[Screenshot 2024-08-23 at 11.47.38 AM.png|400]]
    • ![[Screenshot 2024-08-23 at 11.48.03 AM.png|400]]
  • ReactElement:
    • 组装
    • ![[Screenshot 2024-08-23 at 11.49.34 AM.png|400]]
    • 被装到虚拟 DOM, 然后ReactDOM.render 来填补到真实 DOM: ReactDOM.render(element, container, [callbck])
React 生命周期
  • 灵魂 render
    • 渲染工作流: 从组件数据改变到组件实际更新发生的过程
    • render不会操作真实 DOM, 只是把需要渲染的内容返回出来
    • componentWillReceiveProps()由父组件的更新触发的
    • shouldComponentUpdate()的返回值决定之后的生命周期 & 是否进行 re-render
    • ![[Screenshot 2024-08-23 at 12.51.43 PM.png|500]]
  • 躯干
  • Fiber
    • 把同步变成异步渲染 async render
    • 任务拆解: 把大任务拆解到小任务
    • 可打断: render可以被打断, 并且 by priority 防止饿死

Promises

To handle asynchronous operations

  • asynchronous: they don’t block the main thread of execution
    A Promise is an object that represents the eventual result of an asynchronous operation. It has three states: pendingresolved (fulfilled), and rejected. The core idea is to make asynchronous code look and feel more like synchronous code, making it more readable and maintainable.
  • Promise.resolve(value):This method returns a Promise that is resolved with the given value.
  • Promise.reject(reason):Returns a Promise that is rejected with the given reason.
  • Promise.all(iterable):This method returns a Promise that is resolved when all Promises in the iterable have resolved, or rejected when any of them rejects.
  • Promise.race(iterable):Returns a Promise that is resolved or rejected as soon as one of the Promises in the iterable resolves or rejects.

Axios

make HTTP requests from node.js or XMLHttpRequests from the browser and it supports the Promise API that is native to JS ES6. It can be used intercept HTTP requests and responses and enables client-side protection against XSRF. It also has the ability to cancel requests.

1
2
3
4
5
6
7
8
axios
.get("url")
.then((response) => {
// Code for handling the response
})
.catch((error) => {
// Code for handling the error
});

Hooks

useState
1
const [count, setCount] = useState(0);
useEffect

Each Effect in your code should represent a separate and independent synchronization process.

  • A component mounts when it’s added to the screen.
  • A component updates when it receives new props or state, usually in response to an interaction.
  • A component unmounts when it’s removed from the screen.
  • example
    • components:
      1. ChatRoom mounted with roomId set to "general"
      2. ChatRoom updated with roomId set to "travel"
      3. ChatRoom updated with roomId set to "music"
      4. ChatRoom unmounted
    • Effect:
      1. Your Effect connected to the "general" room
      2. Your Effect disconnected from the "general" room and connected to the "travel" room
      3. Your Effect disconnected from the "travel" room and connected to the "music" room
      4. Your Effect disconnected from the "music" room
1
2
3
4
5
6
7
8
9
10
useEffect(() => {
const connection = createConnection(serverUrl, roomId);

connection.connect();

return () => {
// clean up function to stop synchronizing effect (dependencies changes)
connection.disconnect();
};
}, [roomId]);
createContext
  1. Create the context

    1
    2
    3
    4
    5
    6
    7
    8
    9
    export const PortfolioContext = createContext<{
    data: AllDataType;
    $locale: LocalType;
    onLocaleChange: (locale: LocalType) => void;
    }>({
    $locale: "en-US",
    data: allData,
    onLocaleChange: () => {},
    });
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    export const PortfolioProvider: FC<{ children: ReactNode }> = ({
    children,
    }) => {
    const [$locale, setLocale] = useState < LocalType > "en-US";
    const onLocaleChange = (locale: LocalType) => {
    setLocale(locale);
    };
    return (
    <PortfolioContext.Provider
    value={{ data: allData, $locale, onLocaleChange }}
    >
    {children}
    </PortfolioContext.Provider>
    );
    };
  2. Provide the context (in parent)

    • Wrap children with a context provider to provide the PortfolioProvider
    1
    2
    3
    4
    5
    6
    7
    8
    9
    import { LevelContext } from "./LevelContext.js";

    export default function Section({ level, children }) {
    return (
    <section className="section">
    <PortfolioProvider value={level}>{children}</PortfolioProvider>
    </section>
    );
    }
  3. Use the context (in children)

    • useContext tells React that the component X wants to read the LevelContext.
      const level = useContext(LevelContext);
useReducer

高级版本的 useState

  • 管理复杂的状态逻辑
  • 支持合并多个更新操作,从而减少不必要的重新渲染
  1. interface

    1
    2
    3
    4
    5
    6
    7
    8
    export interface CacheState {
    tab: string
    busy: boolean
    }

    export type CacheAction =
    | { type: "SET"; tab: string }
    | { type: "ADD"; busy?: boolean }
  2. implementation

    • 接收新的状态&动作, 这样就可以更新状态
1
2
3
4
5
6
7
8
export const reducer = (state: CacheState, action: CacheAction): CacheState => {
switch (action.type) {
case "SET":
return { ...state, tab: action.tab };
case "ADD":
return { ...state, busy: action.busy };
}
};
  1. usage
    • useReducer
      • Argument: reducer function, initial state
      • Return: newState, dispatch function (to update state & re-render)
1
2
3
4
5
6
const [{ tab, busy }, dispatch] = useReducer(reducer, {
tab: "Sample",
busy: true,
});

dispatch({ type: "SET", tab: "newTab" });
useMemo

useMemo cache the result of a calculation between re-renders.

  • calculateValue: the Value to cache. Executes for the initial render, then wait until dependencies changes
  • dependencies: The list of all reactive values referenced inside body.
1
2
3
4
import { useMemo } from "react";
function TodoList({ todos, tab }) {
const visibleTodos = useMemo(() => filterTodos(todos, tab), [todos, tab]);
}
useCallback

for referential equality problem

  • 防止因为父组件的非相关渲染而导致的子组件的不必要重新渲染。
  • 想象这个场景:
    • 子组件接受一个父组件传递的函数作为 prop
    • 如果父组件重新渲染,而且这个函数是在父组件的函数体内定义的,那么每次父组件渲染时,都会为子组件传递一个新的函数实例。
    • 这可能会导致子组件不必要地重新渲染,即使该函数的实际内容没有任何变化。
  • 使用场景
    • 子组件的性能优化:当你将函数作为 prop 传递给已经通过 React.memo 进行优化的子组件时,使用 useCallback 可以确保子组件不会因为父组件中的函数重建而进行不必要的重新渲染。
    • Hook 依赖:如果你正在传递的函数会被用作其他 Hook(例如 useEffect)的依赖时,使用 useCallback 可确保函数的稳定性,从而避免不必要的副作用的执行。
    • 复杂计算与频繁的重新渲染:在应用涉及很多细粒度的交互,如绘图应用或其它需要大量操作和反馈的场景,使用 useCallback 可以避免因频繁的渲染而导致的性能问题。
    • 在这个情况中, 除去 number 变化, 重建它是没有意义的. 只有在 number 变化的时候, 才会重建
      1
      2
      3
      4
      const [number, setNumber] = useState(1);
      const getItems = () => {
      return [number, number + 1];
      };
      1
      2
      3
      const getItems = useCallback(() => {
      return [number, number + 1];
      }, [number]);
  • Note
    • useCallback 适用对象是一个 function, 一个被 execute 之后才会 return 的 function
      • useCallback(fn, deps)
    • 我们用 useCallback 把函数包起来之后,在父组件中只有当 deps 变化的时候,才会创建新的 getItems 实例,子组件才会跟着 reRender
    • useMemo 保存的是一个 object, 一个已经被 return 了的 function
      • useMemo(() => fn, deps)
useRef
Ref

ref 是一种用于获取对特定元素或组件实例的引用的方式: React.createRef()

  • 它通常用于访问特定元素,如输入框或自定义组件的实例,或者访问组件上的方法以调用该方法与组件进行交互。
Hook

Useful to persist values across renders - 在 current 属性中保存一个可变的”盒子“
const refContainer = useRef(initialValue);

  • refContainer.current = initialValue
  • 更新 current 不会 re-render