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

相关推荐
crary,记忆1 小时前
Angular微前端架构:Module Federation + ngx-build-plus (Webpack)
前端·webpack·angular·angular.js
漂流瓶jz2 小时前
让数据"流动"起来!Node.js实现流式渲染/流式传输与背后的HTTP原理
前端·javascript·node.js
SamHou02 小时前
手把手 CSS 盒子模型——从零开始的奶奶级 Web 开发教程2
前端·css·web
我不吃饼干2 小时前
从 Vue3 源码中了解你所不知道的 never
前端·typescript
开航母的李大2 小时前
【中间件】Web服务、消息队列、缓存与微服务治理:Nginx、Kafka、Redis、Nacos 详解
前端·redis·nginx·缓存·微服务·kafka
Bruk.Liu2 小时前
《Minio 分片上传实现(基于Spring Boot)》
前端·spring boot·minio
鱼樱前端3 小时前
Vue3+d3-cloud+d3-scale+d3-scale-chromatic实现词云组件
前端·javascript·vue.js
zhangxingchao3 小时前
Flutter入门:Flutter开发必备Dart基础
前端
佚名猫3 小时前
vue3+vite+pnpm项目 使用monaco-editor常见问题
前端·vue3·vite·monacoeditor
满分观测网友z3 小时前
vue的<router-link>的to里面的query和params的区别
前端·javascript·vue.js