javascript动画如何创建_requestAnimationFrame为何更好?

发布时间 - 2026-01-07 00:00:00    点击率:
requestAnimationFrame 更适合动画因其自动对齐浏览器刷新节奏、具备页面不可见时自动暂停的节流能力,且提供高精度时间戳用于精准帧控。

requestAnimationFrame 为什么比 setTimeout 更适合做动画

因为 requestAnimationFrame 会自动对齐浏览器的刷新节奏(通常是每秒 60 帧),而 setTimeout 是基于系统时钟计时,容易出现掉帧、卡顿或“撕裂”。它还具备自动节流能力——页面不可见时(比如切到其他 tab),requestAnimationFrame 会被浏览器暂停,省电又省资源。

常见错误现象:setTimeout(..., 16) 看似想模拟 60fps,但实际执行时机不可控,连续多次调用后误差累积,动画越来越拖慢;更糟的是,用户滚动页面或触发重排时,setTimeout 仍强行执行,造成大量无意义计算。

  • 不要手动算 16ms:浏览器刷新率不总是 60Hz(比如高刷屏是 120Hz,对应约 8.3ms)
  • 不要嵌套多个 setTimeout 模拟循环:这会绕过浏览器的帧调度机制
  • requestAnimationFrame 回调接收一个时间戳参数(DOMHighResTimeStamp),应优先用它计算 delta,而不是依赖 Date.now()

如何正确写一个 requestAnimationFrame 动画循环

核心是“递归调用 + 时间差控制”,不是简单地在回调里再调一次 requestAnimationFrame 就完事。你需要记录上一帧时间,根据当前帧时间差决定是否更新状态,否则动画速度会随帧率浮动(比如在低性能设备上变慢)。

let lastTime = 0;
function animate(currentTime) {
  if (!lastTime) lastTime = currentTime;
  const deltaTime = currentTime - lastTime;
  lastTime = currentTime;

// 例如:让元素每秒移动 100px const speed = 100; // px/s element.style.transform = translateX(${(speed * deltaTime / 1000).toFixed(2)}px);

requestAnimationFrame(animate); } requestAnimationFrame(animate);

注意点:

  • 必须用传入的 currentTime(不是 Date.now()),它是高精度单调递增时间戳
  • 避免在回调中做重排(如读取 offsetTop)或重绘(如频繁改 style),否则触发同步布局,严重拖慢帧率
  • 如果动画需暂停/恢复,保存 requestId 并用 cancelAnimationFrame 控制

requestAnimationFrame 和 CSS 动画该选哪个

优先用 CSS 动画(transition@keyframes),因为它们由浏览器渲染线程直接驱动,不经过 JS 主线程,性能更高、更省电。JS 的 requestAnimationFrame 只应在需要动态计算(比如跟随鼠标、物理模拟、滚动联动)时使用。

典型误用场景:

  • 只做简单位移/缩放/透明度变化,却用 JS + requestAnimationFrame 手动更新 style → 应改用 transition
  • requestAnimationFrame 中反复读写同一元素的 offsetLeftstyle.left → 强制同步布局,直接干掉 60fps
  • requestAnimationFrame 驱动 SVG transform 却没开启 will-change 或 contain → 浏览器无法启用 GPU 加速

兼容性与降级处理要不要做

现代浏览器(Chrome 24+、Firefox 23+、Safari 6.1+、Edge 10+)都支持 requestAnimationFrame,IE10+ 有带前缀的 msRequestAnimationFrame,但 IE 已退出主流支持。除非明确要兼容 IE9 及更早版本,否则无需降级到 setTimeout

如果真要兼容,可用这个轻量 fallback:

const raf = window.requestAnimationFrame ||
            window.webkitRequestAnimationFrame ||
            window.mozRequestAnimationFrame ||
            window.msRequestAnimationFrame ||
            function(callback) {
              return setTimeout(callback, 1000 / 60);
            };

但要注意:这个 fallback 在低帧率设备或后台 tab 中不会自动降频,可能持续消耗 CPU。真正关键的动画逻辑(比如游戏、可视化)建议直接放弃 IE9–。

最常被忽略的一点:很多人以为只要用了 requestAnimationFrame 就一定丝滑,其实它只是“调度器”,动画是否流畅,最终取决于每一帧内做的操作是否轻量——比如 layout thrashing、强制同步重排、未优化的 canvas 绘制,都会让 requestAnimationFrame 失效。


# css  # javascript  # java  # js  # svg  # 浏览器  # edge  # safari  # ai  # win  # 重绘  # 为什么 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: linux top下的 minerd 木马清除方法  LinuxShell函数封装方法_脚本复用设计思路【教程】  PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】  如何快速重置建站主机并恢复默认配置?  Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】  Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】  html5audio标签播放结束怎么触发事件_onended回调方法【教程】  Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程  Laravel Livewire是什么_使用Laravel Livewire构建动态前端界面  香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧  Laravel怎么定时执行任务_Laravel任务调度器Schedule配置与Cron设置【教程】  Laravel如何使用Service Provider注册服务_Laravel服务提供者配置与加载  Laravel的辅助函数有哪些_Laravel常用Helpers函数提高开发效率  Swift中循环语句中的转移语句 break 和 continue  Swift开发中switch语句值绑定模式  香港网站服务器数量如何影响SEO优化效果?  javascript读取文本节点方法小结  JavaScript如何实现倒计时_时间函数如何精确控制  Laravel路由怎么定义_Laravel核心路由系统完全入门指南  微信h5制作网站有哪些,免费微信H5页面制作工具?  油猴 教程,油猴搜脚本为什么会网页无法显示?  C语言设计一个闪闪的圣诞树  谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  如何在IIS管理器中快速创建并配置网站?  EditPlus 正则表达式 实战(3)  如何在万网自助建站平台快速创建网站?  JS中页面与页面之间超链接跳转中文乱码问题的解决办法  标题:Vue + Vuex 项目中正确使用 JWT 进行身份认证的实践指南  Python文件异常处理策略_健壮性说明【指导】  Laravel怎么写单元测试_PHPUnit在Laravel项目中的基础测试入门  Laravel如何使用Guzzle调用外部接口_Laravel发起HTTP请求与JSON数据解析【详解】  文字头像制作网站推荐软件,醒图能自动配文字吗?  laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析  C++用Dijkstra(迪杰斯特拉)算法求最短路径  图册素材网站设计制作软件,图册的导出方式有几种?  实例解析angularjs的filter过滤器  今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】  太平洋网站制作公司,网络用语太平洋是什么意思?  电视网站制作tvbox接口,云海电视怎样自定义添加电视源?  Gemini手机端怎么发图片_Gemini手机端发图方法【步骤】  jquery插件bootstrapValidator表单验证详解  极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?  济南网站建设制作公司,室内设计网站一般都有哪些功能?  微信小程序 HTTPS报错整理常见问题及解决方案  再谈Python中的字符串与字符编码(推荐)  Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程  Laravel如何优化应用性能?(缓存和优化命令)  如何注册花生壳免费域名并搭建个人网站?  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议