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 或状态管理库。

相关推荐
白兰地空瓶10 分钟前
🚀你以为你在写 React?其实你在“搭一套前端操作系统”
前端·react.js
爱上妖精的尾巴44 分钟前
6-4 WPS JS宏 不重复随机取值应用
开发语言·前端·javascript
似水流年QC1 小时前
深入探索 WebHID:Web 标准下的硬件交互实现
前端·交互·webhid
陪我去看海1 小时前
测试 mcp
前端
speedoooo2 小时前
在现有App里嵌入一个AI协作者
前端·ui·小程序·前端框架·web app
全栈胖叔叔-瓜州2 小时前
关于llamasharp 大模型多轮对话,模型对话无法终止,或者输出角色标识User:,或者System等角色标识问题。
前端·人工智能
三七吃山漆2 小时前
攻防世界——wife_wife
前端·javascript·web安全·网络安全·ctf
用户47949283569152 小时前
面试官问"try-catch影响性能吗",我用数据打脸
前端·javascript·面试
GISer_Jing3 小时前
前端营销技术实战:数据+AI实战指南
前端·javascript·人工智能
GIS之路3 小时前
使用命令行工具 ogr2ogr 将 CSV 转换为 Shp 数据(二)
前端