
本文提供一种基于预加载全部帧图像并消除 img.src 动态赋值竞态条件的 canvas 动画优化方案,彻底解决因图片未就绪导致的闪烁、空白帧和卡顿问题。
在使用 Canvas 实现基于滚动触发的逐帧动画(如产品渲染图轮播)时,常见的“画面突然变透明”或“短暂闪白”现象,往往并非源于 clearRect() 或 drawImage() 性能瓶颈,而是由图像资源加载的异步性与绘制时机不匹配引发的竞态条件(race condition):当代码执行 img.src = nextFrameUrl 后立即调用 drawImage(img, ...),而此时新图片尚未完成加载(onload 未触发),img 仍处于空/无效状态,导致 Canvas 绘制空白。
原始代码中仅预加载图片但未保存引用,且每次滚动都复用同一个 元素并动态修改其 src,这极易造成绘制时图像未就绪——尤其在网络波动或首屏未缓存场景下更为明显。
✅ 正确解法是:预加载所有帧为已就绪的 Image 实例,并缓存在数组中,绘制时直接复用已加载完成的对象。
以下是优化后的完整实现:
const canvas = document.getElementById("prodPicAnimation");
const ctx = canvas.getContext("2d");
const frameCount = 120;
// 生成全部帧 URL(1-based,补零至3位)
const imageUrls = Array.from({ length: frameCount }, (_, i) =>
`https://www.dr-adler.com/content/produkte/Sheabutter/renderAnimation/${String(i + 1).padStart(3, '0')}.png`
);
// 封装图片预加载 Promise
const preloadImage = (src) => {
return new Promise((resolve) => {
const img = new Image();
img.onload = () => resolve(img);
img.onerror = () => console.warn(`Failed to load frame: ${src}`);
img.src = src;
});
};
// 并行预加载所有帧,返回已就绪 Image 实例数组
const preloadAllFrames = () => Promise.all(imageUrls.map(preloadImage));
// 初始化:设置画布尺寸、预加载、绑定滚动逻辑
const init = async () => {
canvas.width = 750;
canvas.height = 750;
// 显示加载状态(可选)
const loadingEl = document.getElementById('loading');
if (loadingEl) loadingEl.textContent = 'Loading frames...';
try {
const frames = await preloadAllFrames();
console.log(`✅ All ${frameCount} frames loaded successfully.`);
// 移除加载提示(如有)
if (loadingEl) loadingEl.remove();
// 初始绘制第一帧
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.drawImage(frames[0], 0, 0);
// 滚动监听:根据滚动位置映射到对应帧索引,并 requestAnimationFrame 安全绘制
window.addEventListener('scroll', () => {
const scrollTop = document.documentElement.scrollTop;
const maxScrollTop = 750; // 与动画总高度对齐
const scrollFraction = Math.max(0, Math.min(1, scrollTop / maxScrollTop));
const frameIndex = Math.min(frameCount - 1, Math.floor(scrollFraction * frameCount));
requestAnimationFrame(() => {
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.drawImage(frames[frameIndex], 0, 0);
});
});
} catch (err) {
console.error('❌ Frame preloading failed:', err);
}
};
// 启动
init();? 关键优化点说明:
- ✅ 无竞态:每帧 Image 对象在 onload 后才进入数组,drawImage 调用时 100% 确保图像已就绪;
- ✅ 零延迟切换:无需等待 src 赋值 → 加载 → 绘制的链路,直接复用内存中已解码图像;
- ✅ 内存友好:现代浏览器对已解码图像有良好缓存,120 张 PNG(若单张 ≤200KB)通常占用可控内存(约 20–50MB);
- ✅ 滚动解耦:requestAnimationFrame 确保绘制与屏幕刷新率同步,避免丢帧;
- ✅ 健壮性增强:添加 onerror 处理、边界校验(Math.max/min)、加载失败兜底。
⚠️ 注意事项:
- 若帧数极多(如 >300)或单帧体积过大(>500KB),建议结合 分片懒加载 + LRU 缓存策略,避免初始内存峰值;
- 确保服务端对 PNG 启用 Cache-Control: public, max-age=31536000,利用强缓存减少重复请求;
- 在 drawImage 前始终调用 clearRect() —— 即使图像带透明通道,残留像素也可能因混合模式导致视觉异常。
通过将“加载”与“绘制”彻底分离,并以 Promise 驱动预加载流程,该方案从根本上消除了帧动画的闪烁根源,实现在 Chrome/Firefox/Safari 中稳定、60fps 的透明逐帧滚动动画。










