面试官又问我是否了解React的单向数据流

"React 的单向数据流让数据变化更可预测,这是 React 设计的基石。"

介绍

单向数据流:从字面意思我们就可以理解只能从一端到另一端,而不是反过来。

在React中的表现就是,数据在组件树中自上而下传递,从父组件流向子组件。

单向传递

js 复制代码
// 父组件传递数据
function Parent() {
  const [count, setCount] = useState(0);
  return <Child count={count} />;
}

// 子组件接收只读props
function Child({ count }) {
  return <div>{count}</div>;
}

子组件如何影响父组件

另一方面,父组件可以通过props 将数据传递给子组件,子组件不能直接修改父组件传递过来的props,只能通过触发父组件传递的回调函数来间接更新父组件的状态。

js 复制代码
function Parent() {
  const [count, setCount] = useState(0);
  const handleIncrement = () => setCount(c => c + 1);
  return <Child onIncrement={handleIncrement} />;
}

function Child({ onIncrement }) {
  return <button onClick={onIncrement}>+1</button>;
}

这样的设计的好处就是自上而下的数据流动,使得组件之间的关系更加清晰,数据变化更容易追踪。

如果随便一个子组件都能修改父组件的数据,那数据就会变得不可控,一旦出了问题,都难以查找。

优点

简单总结一下单向数据流的好处:

  • 解耦:子组件不直接依赖父组件,仅通过props接口通信
  • 可预测性:组件间的通信更为清晰,出了问题直接往props中找。

跨组件通信

可能看到这里,就会有小伙伴发现一个问题,拿要是组件嵌套n层,props岂不是要传递n层,一层一层传下来? emm 也不是不可以。当然 React 也意识到了这个问题,所以给我们提供了 context 这个玩意。

Context 让父组件可以为它下面的整个组件树提供数据。

用法也很简单:

  • 先创建 Context (createContext);
  • 再传递 (Provider);
  • 再消费(useContext)
js 复制代码
import { createContext, useContext, useState } from 'react';

// 创建 Context
const UserContext = createContext({
  user: null,
  login: () => {},
  logout: () => {},
});

function App() {
  const [user, setUser] = useState(null);

  const login = (name) => setUser({ name });
  const logout = () => setUser(null);

  return (
    <UserContext.Provider value={{ user, login, logout }}>
      <AuthButton />
    </UserContext.Provider>
  );
}

function AuthButton() {
  const { user, login, logout } = useContext(UserContext);
  
  return user ? (
    <button onClick={logout}>退出登录(当前用户:{user.name})</button>
  ) : (
    <button onClick={() => login('张三')}>登录</button>
  );
}

不要滥用

每当 Providervalue 变化时,所有使用这个 Context 的子组件都会重新渲染!!!

所以当我们使用Context之前,应该先问问自己 能不能先使用props,

使用场景

React 官网也给我们举了几个例子:

  • 主题切换
  • 存储账号信息:许多组件可能需要知道当前登录的用户信息
  • 路由:比如我们需要知道当前是那个路由,需要高亮。
  • 将 reducer 与 context 搭配使用来管理复杂的状态。

留个思考:这时候可能面试官会问你,既然是单向数据流,那怎么实现逆向通信呢??

相关推荐
竹林8181 小时前
用 wagmi v2 + viem 监听链上事件,我踩了三天坑终于搞懂了实时日志与历史补全
javascript
Momo__1 小时前
VueUse createReusableTemplate —— 单文件组件内的模板复用神器
前端·vue.js
无名氏同学1 小时前
React 16-19 新特性
react.js
只一1 小时前
😭从回调地狱到 async/await:一文打通 Ajax 与 JS 异步编程
javascript
程序员小富1 小时前
我开源了一个开发者专属的智能 JSON 工具,得到了媳妇高度认可
前端·vue.js·后端
小小小小宇1 小时前
程序员如何给 LLM 装工具以及看懂推理过程
前端
写代码的皮筏艇1 小时前
React中的forwardRef
前端·react.js·面试
槑有老呆1 小时前
花三个月工资请了个 AI 程序员,结果它连青岛啤酒股价都查不了
前端
风骏时光牛马1 小时前
Verilog开发常见问题汇总解析
前端
子兮曰1 小时前
AI Coding Method Map:一张图看懂 AI 编程的完整链路
前端·人工智能·后端