对useEffect和 useMemo的一些总结与感悟

今天在项目中遇到useEffect和useMemo的一些坑,总结一下。

在现代的react框架编程中,需要摒弃使用Ref来进行状态管理。

如果在一个组件中,重复进行ReactQuery的Query的请求,需要检查其QueryKey是否重合。如:

javascript 复制代码
export function useGetUserWithName(username: string) {
  return useQuery({
    queryKey: ['user', 'data', username],
    queryFn: async () => {
      const res = await http.get<GetUserWithNameApiResponse>(
        getUserWithNamePath + `/${username}`
      );
      return res;
    },
    enabled: !!username,
  });
}

export function useGetAvatar(username: string) {
  return useQuery({
    queryKey: ['user', 'avatar', username],
    queryFn: async () => {
      const res = await http.get<GetUserAvatarApiResponse>(
        GetUserAvatarPath + `/${username}`
      );
      return res;
    },
    enabled: !!username,
  });
}

上面的两个queryKey如果重合,数据则会被先请求的覆盖掉。所以加第二个缓存标识很有必要。

还有enabled,必须要加。加上后在组件中调用接口的时候,就不会陷入死循环,重复去一直调用接口。

那么以上和useEffect、useMemo有什么关系呢?

当使用接口时,必然会有:

kotlin 复制代码
  const { data: avatarResponse } = useGetAvatar(username);
  const { data: userdata } = useGetUserWithName(username);

这时它们都在组件的宏任务中,但是它们的function都是async的,所以宏任务依次调用后会添加到它们自己的异步任务中。

然后就是返回时机,它们返回的时机不一致,会导致数据来的时间不一致。

如果要处理其中的数据,则需要对其进行状态管理。

那么第一步就是拿出里面的数据。而我们知道,一个接口的数据返回出来,可能在深度上太深,所以我们要处理深度,避免直接调它们的data。

那么如何处理深度?

ini 复制代码
const userinfo = useMemo((): GetUserWithNameApiResponse | undefined => {
    return userdata?.data;
}, [userdata]);

这样就会减少数据的深度,我们也能更清晰的拿到数据。

kotlin 复制代码
const avatarUrl = useMemo(() => {
    if (!avatarResponse?.data?.data?.buffer?.data) {
      return undefined;
    }

    try {
      // 提取 buffer 数据(数字数组)
      const bufferData = avatarResponse.data.data.buffer.data;
      const mimeType = avatarResponse.data.data.mimetype || 'image/jpeg';

      // 将数字数组转换为 Uint8Array
      const uint8Array = new Uint8Array(bufferData);

      // 创建 Blob
      const blob = new Blob([uint8Array], { type: mimeType });

      // 创建对象 URL
      return URL.createObjectURL(blob);
    } catch (error) {
      console.error('Failed to convert avatar buffer to URL:', error);
      return undefined;
    }
  }, [avatarResponse]);

上述代码是将图片的数据结构取出,转换为Blob形式。其实也可以转为Base64的数据字符串。

那么useMemo就是进行这种运算的。它会将数据进行处理。

那么useEffect呢?它其实就是处理数据的副作用的。

kotlin 复制代码
useEffect(() => {
    // 只在数据真正改变时才更新,避免无限循环
    if (!username) {
      return;
    }

    // eslint-disable-next-line react-hooks/set-state-in-effect
    setProfile((prev) => ({
      nickname: userinfo?.data.username || username || '',
      email: userinfo?.data.email || prev.email || '',
      phone: userinfo?.data.phone || prev.phone || '',
      bio: userinfo?.data.desc || prev.bio || '',
      avatar: avatarUrl || prev.avatar || null,
    }));
  }, [username, avatarUrl, userinfo]);

可以看到,使用useEffect对数据读取并设置值,这就是数据的"作用"。 需要注意的是,最好在一个数据结构更新时设置一个useEffect统一更新数据,不然会有数据不一致的问题。

当A数据是H结构的一份子,B也是H结构的一份子时,如果使用两个useEffect去同时修改H,因为A和B获取数据是async的,所以哪一个先到先第一个覆盖数据。

当A先到,B后到时,B就会覆盖A的所有数据。因为H结构不能有空值,但是B数据又没有A数据,所以只能设置默认值,这样设置的默认值会把先前A设置的数据覆盖。

所以需要一个useEffect同时设定A和B。上面的代码块就是一个例子。

相关推荐
fanruitian2 小时前
uniapp android开发 测试板本与发行版本
前端·javascript·uni-app
rayufo2 小时前
【工具】列出指定文件夹下所有的目录和文件
开发语言·前端·python
RANCE_atttackkk2 小时前
[Java]实现使用邮箱找回密码的功能
java·开发语言·前端·spring boot·intellij-idea·idea
2501_944525544 小时前
Flutter for OpenHarmony 个人理财管理App实战 - 支出分析页面
android·开发语言·前端·javascript·flutter
李白你好4 小时前
Burp Suite插件用于自动检测Web应用程序中的未授权访问漏洞
前端
刘一说5 小时前
Vue 组件不必要的重新渲染问题解析:为什么子组件总在“无故”刷新?
前端·javascript·vue.js
徐同保6 小时前
React useRef 完全指南:在异步回调中访问最新的 props/state引言
前端·javascript·react.js
刘一说7 小时前
Vue 导航守卫未生效问题解析:为什么路由守卫不执行或逻辑失效?
前端·javascript·vue.js
一周七喜h7 小时前
在Vue3和TypeScripts中使用pinia
前端·javascript·vue.js
weixin_395448917 小时前
main.c_cursor_0202
前端·网络·算法