react路由中使用context

Context(类似 Vue 的 provide/inject,适合中等规模)

复制代码
// 创建 Context
const ThemeContext = React.createContext<'light' | 'dark'>('light');

// 提供者
function App() {
  const [theme, setTheme] = useState<'light'|'dark'>('light');
  return (
    <ThemeContext.Provider value={theme}>
      <Toolbar />
    </ThemeContext.Provider>
  );
}

// 消费者(在任意子组件)
function Toolbar() {
  const theme = useContext(ThemeContext);
  return <div>Current theme: {theme}</div>;
}

详细介绍一下context 使用方法

第一步:将 Context 定义在一个单独的文件(或至少导出)

复制代码
// contexts/ThemeContext.tsx
import { createContext } from 'react';

export const ThemeContext = createContext<'light' | 'dark'>('light');

第二步:在 App 中导入 Provider 并包裹路由

复制代码
// App.tsx
import { BrowserRouter, Routes, Route, Link } from 'react-router-dom';
import { ThemeContext } from './contexts/ThemeContext';
import Home from './pages/home';
import About from './pages/About';
import UserDetail from './pages/userDetail';

function App() {
  const [theme, setTheme] = useState<'light'|'dark'>('light');
  return (
    <BrowserRouter>
      <ThemeContext.Provider value={theme}>
        <nav>
          <Link to="/">Home</Link>
          <Link to="/about">About</Link>
        </nav>
        <Routes>
          <Route path="/" element={<Home />} />
          <Route path="/about" element={<About />} />
          <Route path="/user/:id" element={<UserDetail />} />
        </Routes>
      </ThemeContext.Provider>
    </BrowserRouter>
  );
}

第三步:在任意路由组件中导入并使用同一个 Context

复制代码
// pages/home.tsx
import { useContext } from 'react';
import { ThemeContext } from '../contexts/ThemeContext';

export default function Home() {
  const theme = useContext(ThemeContext);
  return <div>Home - current theme: {theme}</div>;
}

// pages/userDetail.tsx
import { useContext } from 'react';
import { useParams } from 'react-router-dom';
import { ThemeContext } from '../contexts/ThemeContext';

export default function UserDetail() {
  const { id } = useParams();
  const theme = useContext(ThemeContext);
  return (
    <div>
      User ID: {id} <br />
      Theme: {theme}
    </div>
  );
}

常见错误与检查清单

错误现象 可能原因 解决方法
useContext(ThemeContext) 返回默认值 'light',修改 Provider 的 value 也没用 你在组件中导入的 ThemeContext 和 Provider 使用的不是同一个对象(比如从不同文件导出了两个不同的 Context) 确保整个应用只定义一次 createContext,并且所有地方都从同一个文件导入
报错 ThemeContext is not defined 没有导入 Context 在需要使用 useContext 的文件里 import { ThemeContext } from '...'
Context 值变化了,但某些路由组件没有更新 可能是路由组件被 React.memo 包裹,或者你在 Provider 外面使用了 BrowserRouter 导致 Context 没有包裹到该路由 ThemeContext.Provider 放在 BrowserRouter 内部 (但包裹住 Routes 就行,你原来的位置是对的)
UserDetail 组件中 useParams() 拿不到 id 路由配置错误:<Route path="/user/:id" element={<UserDetail />} /> 必须正确,且 UserDetail 必须是在 <Routes> 内渲染的组件,不能是嵌套在 App 内部的函数 UserDetail 提取为单独的文件或至少定义在 App 外部

总结

  • Context 在路由组件 ,只要 Provider 包裹了 <Routes>

  • 关键点:同一个 Context 对象必须被所有使用它的组件导入。

  • 如果你按文件拆分,记得 exportimport

相关推荐
Hilaku2 小时前
一周狂揽40K+ Star⭐ 的 Pretext 到底有多变态?
前端·javascript·html
前端郭德纲2 小时前
JavaScript 原型相关属性详解
开发语言·javascript·原型模式
533_2 小时前
适用于vue3的拖拽插件:vue-draggable-plus, vuedraggable@next
javascript·vue.js
Southern Wind2 小时前
AI Skill Server 动态技能中台
前端·后端·mysql·node.js
锦木烁光2 小时前
多端项目太乱?我是这样用 Monorepo 重构的
前端·架构
早點睡3902 小时前
ReactNative项目OpenHarmony三方库集成实战:react-native-shadow-2
javascript·react native·react.js
上山打牛2 小时前
cornerstone3D基本使用
前端
阿鑫_9962 小时前
通用-Nvm基础知识
前端
xinzheng新政3 小时前
Javascript·深入学习基础知识
前端·javascript·学习