如何优雅的避免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后打断。

相关推荐
小小愿望几秒前
移动端浏览器中设置 100vh 却出现滚动条?
前端·javascript·css
fail_to_code1 分钟前
请不要再只会回答宏任务和微任务了
前端
摸着石头过河的石头2 分钟前
taro3.x-4.x路由拦截如何破?
前端·taro
lpfasd12311 分钟前
开发Chrome/Edge插件基本流程
前端·chrome·edge
练习前端两年半1 小时前
🚀 Vue3 源码深度解析:Diff算法的五步优化策略与最长递增子序列的巧妙应用
前端·vue.js
烛阴1 小时前
TypeScript 接口入门:定义代码的契约与形态
前端·javascript·typescript
掘金安东尼1 小时前
使用自定义高亮API增强用户‘/’体验
前端·javascript·github
参宿72 小时前
electron之win/mac通知免打扰
java·前端·electron
石小石Orz2 小时前
性能提升60%:前端性能优化终极指南
前端·性能优化
夏日不想说话2 小时前
API请求乱序?深入解析 JS 竞态问题
前端·javascript·面试