react中通过 EventEmitter 在组件间传递状态

要在 Reply 组件中通过 statusChangeEvent 发送状态值,并在 Select 组件中接收这个状态值 status,你可以按照以下步骤实现:

复制代码
//Event.js

import EventEmitter from 'events';

export const statusChangeEvent = new EventEmitter();// 工单状态切换事件

1. 在 Reply 组件中发送状态值

你已经正确地在 Reply 组件中设置了事件发射:

复制代码
// Reply.jsx
import { statusChangeEvent } from './Event';

const Reply = () => {
  const [status, setStatus] = useState('');
  
  const changeStatus = useCallback(status => {
    setStatus(status);
    statusChangeEvent.emit('status-changedValue', status);
  }, []);

  return (
    // 你的组件内容
    <button onClick={() => changeStatus('new-status')}>更改状态</button>
  );
};

2. 在 Select 组件中接收状态值

复制代码
// Select.jsx
import { useEffect, useState } from 'react';
import { statusChangeEvent } from './Event';

const Select = () => {
  const [status, setStatus] = useState('');

  useEffect(() => {
    // 定义事件监听器
    const handleStatusChange = (newStatus) => {
      setStatus(newStatus);
    };

    // 添加事件监听
    statusChangeEvent.on('status-changedValue', handleStatusChange);

    // 组件卸载时移除监听
    return () => {
      statusChangeEvent.off('status-changedValue', handleStatusChange);
    };
  }, []);

  return (
    <div>
      <p>当前状态: {status}</p>
      {/* 其他组件内容 */}
    </div>
  );
};

3. 优化方案(推荐)

虽然 EventEmitter 可以工作,但在 React 中更推荐使用 Context 或状态管理库(如 Redux)来处理跨组件状态共享。以下是使用 Context 的替代方案:

创建状态上下文

复制代码
// StatusContext.js
import { createContext, useContext, useState } from 'react';

const StatusContext = createContext();

export const StatusProvider = ({ children }) => {
  const [status, setStatus] = useState('');
  
  return (
    <StatusContext.Provider value={{ status, setStatus }}>
      {children}
    </StatusContext.Provider>
  );
};

export const useStatus = () => useContext(StatusContext);

在 Reply 组件中使用

复制代码
// Reply.jsx
import { useStatus } from './StatusContext';

const Reply = () => {
  const { setStatus } = useStatus();
  
  const changeStatus = useCallback(status => {
    setStatus(status);
  }, [setStatus]);

  return (
    <button onClick={() => changeStatus('new-status')}>更改状态</button>
  );
};

在 Select 组件中使用

复制代码
// Select.jsx
import { useStatus } from './StatusContext';

const Select = () => {
  const { status } = useStatus();
  
  return (
    <div>
      <p>当前状态: {status}</p>
    </div>
  );
};

在应用顶层包裹 Provider

复制代码
// App.js
import { StatusProvider } from './StatusContext';

function App() {
  return (
    <StatusProvider>
      <Reply />
      <Select />
    </StatusProvider>
  );
}

注意事项

  1. EventEmitter 方案

    • 适合非父子组件间的通信

    • 记得在组件卸载时移除监听器

    • 可能导致难以追踪的数据流

  2. Context 方案

    • React 推荐的方式

    • 更易于维护和调试

    • 当状态变化时,所有使用该状态的组件都会重新渲染

  3. 性能考虑

    • 如果状态更新频繁,考虑使用 useMemo 或 React.memo 优化性能

    • 对于复杂应用,Redux 或 Zustand 可能是更好的选择

选择哪种方案取决于你的应用复杂度和组件结构。对于简单场景,EventEmitter 足够;对于更复杂的应用,推荐使用 Context 或状态管理库。

相关推荐
找不到工作的菜鸟4 分钟前
Three.js三大组件:场景(Scene)、相机(Camera)、渲染器(Renderer)
前端·javascript·html
定栓7 分钟前
vue3入门-v-model、ref和reactive讲解
前端·javascript·vue.js
专注API从业者11 分钟前
基于 Flink 的淘宝实时数据管道设计:商品详情流式处理与异构存储
大数据·前端·数据库·数据挖掘·flink
龙在天12 分钟前
H5开发,开发照相机,以及组件封装
前端
曼妥思19 分钟前
PosterKit:跨框架海报生成工具
前端·开源
binqian38 分钟前
【异步】js中异步的实现方式 async await /Promise / Generator
开发语言·前端·javascript
Jerry说前后端1 小时前
Android 移动端 UI 设计:前端常用设计原则总结
android·前端·ui
熊猫钓鱼1 小时前
基于Trae CN与TrendsHub快速实现的热点百事通
前端·trae
LIUENG1 小时前
Vue3 响应式原理
前端·vue.js
讨厌吃蛋黄酥1 小时前
前端居中九种方式血泪史:面试官最爱问的送命题,我一次性整明白!
前端·css