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

相关推荐
宠..11 分钟前
创建标签控件
java·服务器·开发语言·前端·c++·qt
Z_Wonderful12 分钟前
主题切换(1):css变量的使用(:root)
前端·javascript·css
亮子AI14 分钟前
【CSS】如何选择父级元素?
前端·css
Z_Wonderful15 分钟前
主题切换(2):CSS 变量(自定义属性)
前端·javascript·css
m0_7400437319 分钟前
html练习题
开发语言·前端·javascript
qq_4179165320 分钟前
HTML中的列表
android·前端·html
yqcoder23 分钟前
Vue2 和 Vue3 中祖先组件和子孙组件的通信方法和区别
前端·javascript·vue.js
勇气要爆发24 分钟前
问:当服务器资源有限,前端项目高并发优化策略
前端·性能优化
鹏多多24 分钟前
前端组件二次封装实战:Vue+React基于Element UI/AntD的高效封装策略
前端·vue.js·react.js