React

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 bysuper(props)
- if we don’t need
this.props
, then no need forsuper(props)
- if
super()
, thenthis.props = undefined
4. Components
4.1 Class Component
State & lifecycle methods manageable but not friendly for new code (render components in JSX style)
1 | class component extends React.Component { |
4.2 Function Component
- 创建 Component 可以通过
React.createElement(..);
被 babel 转化成React.createClass
- 适合无状态 / use Hooks
- Manage State & lifecycle methods by hooks (return react component)
1 | function MyComponent(props: Props) { |
4.3 区别 - state
Function Component 需要 useState
管理状态
4.4 区别 - Lifecycle
Function Component 不存在生命周期, 因为 hook 都继承于R.C
, 但可以用useEffect
代替生命周期的作用 -> componentDidMount
& componentWillUnmount
4.5 区别 - 调用方式
1 | const res = functionComponent(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 合成事件和原生事件的处理顺序
- 真实 DOM 元素触发后: 先 document 对象 (原生事件
componentDidMount
), 再 React 事件 (onClick
) - 执行 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 | // slow |
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
inwebpack
配置文件 - 不方便动态修改样式, 需要用内联方式编写
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
- boolean: 通过比较
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 | // parent |
10.3 Between Siblings
父组件作为中间层
1 | <SiblingA 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 | // this = Window |
11.2 Function scope
this
取决于 function 如何被 call
explicitly set:
call
,apply
,bind
the value ofthis
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
14const 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 | const 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
7const f = function () {
return this;
}; // Window
const f = function () {
"use strict";
return this;
}; // undefined
- if not set,
11.3 Object context
this refer to the object to itself.
1 | const object = { |
底层逻辑
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: pending, resolved (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 givenvalue
.Promise.reject(reason)
:Returns a Promise that is rejected with the givenreason
.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 | axios |
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:
ChatRoom
mounted withroomId
set to"general"
ChatRoom
updated withroomId
set to"travel"
ChatRoom
updated withroomId
set to"music"
ChatRoom
unmounted
- Effect:
- Your Effect connected to the
"general"
room - Your Effect disconnected from the
"general"
room and connected to the"travel"
room - Your Effect disconnected from the
"travel"
room and connected to the"music"
room - Your Effect disconnected from the
"music"
room
- Your Effect connected to the
- components:
1 | useEffect(() => { |
createContext
Create the context
1
2
3
4
5
6
7
8
9export 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
15export 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>
);
};Provide the context (in parent)
- Wrap children with a context provider to provide the
PortfolioProvider
1
2
3
4
5
6
7
8
9import { LevelContext } from "./LevelContext.js";
export default function Section({ level, children }) {
return (
<section className="section">
<PortfolioProvider value={level}>{children}</PortfolioProvider>
</section>
);
}- Wrap children with a context provider to provide the
Use the context (in children)
useContext
tells React that the component X wants to read theLevelContext
.const level = useContext(LevelContext);
useReducer
高级版本的 useState
- 管理复杂的状态逻辑
- 支持合并多个更新操作,从而减少不必要的重新渲染
interface
1
2
3
4
5
6
7
8export interface CacheState {
tab: string
busy: boolean
}
export type CacheAction =
| { type: "SET"; tab: string }
| { type: "ADD"; busy?: boolean }implementation
- 接收新的状态&动作, 这样就可以更新状态
1 | export const reducer = (state: CacheState, action: CacheAction): CacheState => { |
- usage
- useReducer
- Argument: reducer function, initial state
- Return: newState, dispatch function (to update state & re-render)
- useReducer
1 | const [{ tab, busy }, dispatch] = useReducer(reducer, { |
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 changesdependencies
: The list of all reactive values referenced inside body.
1 | import { useMemo } from "react"; |
useCallback
for referential equality problem
- 防止因为父组件的非相关渲染而导致的子组件的不必要重新渲染。
- 想象这个场景:
- 子组件接受一个父组件传递的函数作为 prop
- 如果父组件重新渲染,而且这个函数是在父组件的函数体内定义的,那么每次父组件渲染时,都会为子组件传递一个新的函数实例。
- 这可能会导致子组件不必要地重新渲染,即使该函数的实际内容没有任何变化。
- 使用场景
- 子组件的性能优化:当你将函数作为 prop 传递给已经通过 React.memo 进行优化的子组件时,使用 useCallback 可以确保子组件不会因为父组件中的函数重建而进行不必要的重新渲染。
- Hook 依赖:如果你正在传递的函数会被用作其他 Hook(例如 useEffect)的依赖时,使用 useCallback 可确保函数的稳定性,从而避免不必要的副作用的执行。
- 复杂计算与频繁的重新渲染:在应用涉及很多细粒度的交互,如绘图应用或其它需要大量操作和反馈的场景,使用 useCallback 可以避免因频繁的渲染而导致的性能问题。
- 在这个情况中, 除去 number 变化, 重建它是没有意义的. 只有在 number 变化的时候, 才会重建
1
2
3
4const [number, setNumber] = useState(1);
const getItems = () => {
return [number, number + 1];
};1
2
3const 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)
- useCallback 适用对象是一个 function, 一个被 execute 之后才会 return 的 function
useRef
Ref
ref 是一种用于获取对特定元素或组件实例的引用的方式: React.createRef()
- 它通常用于访问特定元素,如输入框或自定义组件的实例,或者访问组件上的方法以调用该方法与组件进行交互。
Hook
Useful to persist values across renders - 在 current 属性中保存一个可变的”盒子“const refContainer = useRef(initialValue);
refContainer.current = initialValue
- 更新 current 不会 re-render