react:usecontext使用

useContext返回 调用组件上方最近的 SomeContext.Provider 的 value

如果没有这样的 provider 传递给 createContext 的 defaultValue。返回的值始终是最新的。

如果 context 发生变化,React 会自动重新渲染读取 context 的组件。

  1. <Context.Provider> 需要位于调用 useContext() 的组件 之上。
  2. 使用 memo 来跳过重新渲染并不妨碍子级接收到新的 context 值。

https://codepen.io/pen?\&editors=001

javascript 复制代码
const { createRoot } = ReactDOM;
const ThemeContext = React.createContext('light'); 
// import { createContext, useContext } from 'react';

const {  Button, Flex  } = antd;

// 由于组件A被放置在了<ThemeContext.Provider value="dark">的范围内,
//它会成功地从上下文中获取到ThemeContext的值。因此,组件A不需要直接传递ThemeContext,并且不会因此而报错。
const A = () => {
  const theme = React.useContext(ThemeContext);
  return <div>123{theme}</div>
}


const App = () => (
  <ThemeContext.Provider value="dark">
  <Flex gap="small" wrap="wrap">
    <Button type="primary">Primary Button</Button>
    <Button>Default Button</Button>
    <Button type="dashed">Dashed Butto1n</Button>
    <Button type="text">Text Button</Button>
    <Button type="link">Link Button</Button>
  
  </Flex>
  <A/>
  </ThemeContext.Provider>
);
const ComponentDemo = App;


createRoot(mountNode).render(<ComponentDemo />);

可以传递对象(通过 context 可以传递任何值)

javascript 复制代码
<CurrentUserContext.Provider
      value={{
        currentUser,
        setCurrentUser
      }}
    >
      <Form />
    </CurrentUserContext.Provider>

可以把把 provider 抽离成组件

javascript 复制代码
<MyProviders theme={theme} setTheme={setTheme}>
      <WelcomePanel />
</MyProviders>

function MyProviders({ children, theme, setTheme }) {
  const [currentUser, setCurrentUser] = useState(null);
  return (
    <ThemeContext.Provider value={theme}>
      <CurrentUserContext.Provider
        value={{
          currentUser,
          setCurrentUser
        }}
      >
        {children}
      </CurrentUserContext.Provider>
    </ThemeContext.Provider>
  );
}

可以同时存在多个content

javascript 复制代码
<ThemeContext.Provider value={theme}>
      <CurrentUserContext.Provider
        value={{
          currentUser,
          setCurrentUser
        }}
      >
        <WelcomePanel />
       </CurrentUserContext.Provider>
    </ThemeContext.Provider>

还可以和useReducer结合使用,在大型项目中

覆盖context

javascript 复制代码
const ThemeContext = createContext(null);

export default function MyApp() {
  return (
    <ThemeContext.Provider value="dark">
      <Form />
    </ThemeContext.Provider>
  )
}

function Form() {
  return (
    <Panel title="Welcome">
      <Button>Sign up</Button>
      <Button>Log in</Button>
      <ThemeContext.Provider value="light">
        <Footer />
      </ThemeContext.Provider>
    </Panel>
  );
}

如果context value是有函数等多个成员时,可以通过useCallback和usememo来优化性能

javascript 复制代码
import { useCallback, useMemo } from 'react';

function MyApp() {
  const [currentUser, setCurrentUser] = useState(null);

  const login = useCallback((response) => {
    storeCredentials(response.credentials);
    setCurrentUser(response.user);
  }, []);

  const contextValue = useMemo(() => ({
    currentUser,
    login
  }), [currentUser, login]);

  return (
    <AuthContext.Provider value={contextValue}>
      <Page />
    </AuthContext.Provider>
  );
}
// 根据以上改变,即使 MyApp 需要重新渲染,调用 useContext(AuthContext) 的组件也不需要重新渲染,
// 除非 currentUser 发生了变化。
相关推荐
我命由我123458 分钟前
SVG - SVG 引入(SVG 概述、SVG 基本使用、SVG 使用 CSS、SVG 使用 JavaScript、SVG 实例实操)
开发语言·前端·javascript·css·学习·ecmascript·学习方法
用户479492835691539 分钟前
给客户做私有化部署,我是如何优雅搞定 NPM 依赖管理的?
前端·后端·程序员
C_心欲无痕1 小时前
vue3 - markRaw标记为非响应式对象
前端·javascript·vue.js
qingyun9891 小时前
深度优先遍历:JavaScript递归查找树形数据结构中的节点标签
前端·javascript·数据结构
胡楚昊1 小时前
NSSCTF动调题包通关
开发语言·javascript·算法
熬夜敲代码的小N1 小时前
Vue (Official)重磅更新!Vue Language Tools 3.2功能一览!
前端·javascript·vue.js
90后的晨仔1 小时前
用 Python 脚本一键重命名序列帧图片的名称
前端
辰同学ovo1 小时前
Vue 2 路由指南:从入门到实战优化
前端·vue.js
小彭努力中1 小时前
1.在 Vue 3 中使用 Cesium 快速展示三维地球
前端·javascript·vue.js·#地图开发·#cesium·#vue3
一棵开花的树,枝芽无限靠近你1 小时前
【face-api.js】1️⃣基于Tensorflow.js的人脸识别项目开源项目
javascript·开源·tensorflow·face-api.js