如何优雅的避免cloudinary云图片加载失败?

前言

我在使用cloudinary(一个云图像/视频平台)的时候发现一个潜在问题:有时候获取图片地址的时候初次获取是无法正确获取到的,必须刷新一次或多次才能正确获取到图片,但是浏览器只会加载一次图片,这就会导致图片即使存在也无法正确加载从而被备用图片替换。

补充一下我使用cloudinary的方式:通过cloudinary提供的上传组件实现图片上传,在图片上传的同时记录下当前上传图片的云地址并存储到数据库,每次使用云图片的时候只需要调用这个地址。

正文

下面是个简单的例子,主要关注src部分,user?.image就是数据库中存储的云图片地址,当我们没有拿到的时候就用"/images/person.jpg"替换。

typescript 复制代码
<img
    ref={imageRef}
    src={user?.image || "/images/person.jpg"}
    alt="image"
    className="w-8 h-8 rounded-full mt-[2px]"
/>

那么如何解决前言中的问题呢?

typescript 复制代码
const imageRef = useRef<HTMLImageElement | null>(null);

const handleImageError = (event: React.SyntheticEvent<HTMLImageElement, Event>) => {
  const imgElement = event.currentTarget;
  imgElement.src = user?.image;
};

return (
  <img
    ref={imageRef}
    src={user?.image || "/images/person.jpg"}
    alt="image"
    className="w-8 h-8 rounded-full mt-[2px]"
    onError={handleImageError}
  />
);

这段代码中我们给onError绑定了handleImageError事件,作用就是当user?.image图片获取错误时都会触发onError事件从而多次加载图片,这样我们就可以避免初次加载无法拿到图片的情况。

这样就结束了吗?当然没有,当user?.image一直无法正确获取就会造成无限循环,所以我们要在其基础上进行循环处理:

typescript 复制代码
const imageRef = useRef<HTMLImageElement | null>(null);
const timerRef = useRef<NodeJS.Timeout | null>(null);  // 用于存储定时器

const handleImageError = (event: React.SyntheticEvent<HTMLImageElement, Event>) => {
  const imgElement = event.currentTarget;
  
  // 如果定时器已存在,说明在2s等待时间内
  if (timerRef.current) {
    // 继续尝试原来的操作
    imgElement.src = user?.image;
    return;
  }

  // 设置2s定时器
  timerRef.current = setTimeout(() => {
    // 2s后直接使用备用地址
    if (imgElement) {
      imgElement.src = "/images/person.jpg";
    }
    // 清除定时器引用
    timerRef.current = null;
  }, 2000);

  // 首次错误时也执行原操作
  imgElement.src = user?.image;
};

// 组件卸载时清理定时器
useEffect(() => {
  return () => {
    if (timerRef.current) {
      clearTimeout(timerRef.current);
    }
  };
}, []);

return (
  <img
    ref={imageRef}
    src={user?.image || "/images/person.jpg"}
    alt="image"
    className="w-8 h-8 rounded-full mt-[2px]"
    onError={handleImageError}
  />
);

通过添加定时器,我们可以设定强制赋值备用图片地址的时间,在这里我设置的是2s可以根据各自情况修改。这样设置后,会给模板图片2s的加载时间,2s后打断。

相关推荐
excel1 小时前
ES6 中函数的双重调用方式:fn() 与 fn\...``
前端
可乐爱宅着2 小时前
全栈框架next.js入手指南
前端·next.js
你的人类朋友3 小时前
什么是API签名?
前端·后端·安全
会豪5 小时前
Electron-Vite (一)快速构建桌面应用
前端
中微子5 小时前
React 执行阶段与渲染机制详解(基于 React 18+ 官方文档)
前端
唐某人丶5 小时前
教你如何用 JS 实现 Agent 系统(2)—— 开发 ReAct 版本的“深度搜索”
前端·人工智能·aigc
中微子5 小时前
深入剖析 useState产生的 setState的完整执行流程
前端
遂心_6 小时前
JavaScript 函数参数传递机制:一道经典面试题解析
前端·javascript
小徐_23336 小时前
uni-app vue3 也能使用 Echarts?Wot Starter 是这样做的!
前端·uni-app·echarts
RoyLin6 小时前
TypeScript设计模式:适配器模式
前端·后端·node.js