React基础 第二十八章(将事件与Effect分开)

在React应用中,正确地区分事件处理函数和Effect是编写高效、可维护代码的关键。本文将探讨如何在事件处理和Effect之间做出选择,以及如何将非响应式逻辑从Effect中分离出来,提高应用的响应性和用户体验。

事件处理函数与Effect的区别

事件处理函数是针对用户交互的,例如点击按钮或提交表单。它们只在特定的用户操作触发时执行。相比之下,Effect是响应式的,它们在组件的props或state发生变化时自动执行,以保持与外部系统的同步。

正确的示例代码:

javascript 复制代码
function ChatRoom({ roomId }) {
  const [message, setMessage] = useState('');

  // 事件处理函数:只在用户点击"Send"按钮时执行
  function handleSendClick() {
    sendMessage(message);
  }

  // Effect:用于与聊天室服务器保持连接
  useEffect(() => {
    const connection = createConnection(serverUrl, roomId);
    connection.connect();
    return () => {
      connection.disconnect();
    };
  }, [roomId]);

  // ...
}

错误的示例代码

javascript 复制代码
function ChatRoom({ roomId }) {
  const [message, setMessage] = useState('');

  // 🔴 错误:在Effect中处理用户点击事件
  useEffect(() => {
    const sendButton = document.getElementById('sendButton');
    sendButton.addEventListener('click', () => {
      sendMessage(message);
    });

    return () => {
      sendButton.removeEventListener('click', () => {
        sendMessage(message);
      });
    };
  }, [message]); // 这里的依赖项可能会导致Effect在每次message变化时重新绑定事件
}

在这个错误的示例中,事件处理逻辑被错误地放在了Effect中。这样做有几个问题:

  1. 每次message变化时,都会重新为发送按钮绑定点击事件监听器,这是不必要的,并且可能导致内存泄漏。
  2. 事件处理函数应该直接绑定到元素上,而不是在Effect中进行绑定和解绑。
  3. 在Effect的清理函数中,由于每次都创建了一个新的函数实例,removeEventListener实际上不会移除之前添加的事件监听器。

如何选择使用事件处理函数还是Effect

当你的代码需要响应用户的特定操作时,应该使用事件处理函数。如果代码需要与组件的props或state保持同步,无论用户是否进行了交互,那么应该使用Effect。

使用Effect Event来隔离非响应式逻辑(实验性API,目前react稳定版不能用)

Effect Event是一个特殊的Hook,它允许你在Effect中执行非响应式逻辑,同时能够访问最新的props和state。

技巧示例代码:

javascript 复制代码
const onConnected = useEffectEvent(() => {
  showNotification('Connected!', theme);
});

useEffect(() => {
  const connection = createConnection(serverUrl, roomId);
  connection.on('connected', onConnected);
  connection.connect();
  return () => connection.disconnect();
}, [roomId]); // 不需要将onConnected作为依赖项

在这个例子中,我们希望在连接到聊天室时显示通知,但不希望每次theme变化时都重新连接。因此,我们没有将theme作为Effect的依赖项。

相关推荐
listhi5202 小时前
利用React Hooks简化状态管理
前端·javascript·react.js
一点一木2 小时前
🚀 2025 年 10 月 GitHub 十大热门项目排行榜 🔥
前端·人工智能·github
华仔啊2 小时前
这个Vue3旋转菜单组件让项目颜值提升200%!支持多种主题,拿来即用
前端·javascript·css
非凡ghost3 小时前
Adobe Lightroom安卓版(手机调色软件)绿色版
前端·windows·adobe·智能手机·软件需求
BestAns3 小时前
Postman 平替?这款轻量接口测试工具,本地运行 + 批量回归超实用!
前端
专注前端30年3 小时前
Webpack进阶玩法全解析(性能优化+高级配置)
前端·webpack·性能优化
烛阴4 小时前
Lua世界的基石:变量、作用域与七大数据类型
前端·lua
张拭心4 小时前
“不卷 AI、不碰币、下班不收消息”——Android 知名技术大牛 Jake Wharton 的求职价值观
android·前端·aigc
SoaringHeart4 小时前
Flutter疑难解决:单独让某个页面的电池栏标签颜色改变
前端·flutter
Yeats_Liao4 小时前
Go Web 编程快速入门 13 - 部署与运维:Docker容器化、Kubernetes编排与CI/CD
运维·前端·后端·golang