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 = c
urrentTime - 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中反复读写同一元素的offsetLeft和style.left→ 强制同步布局,直接干掉 60fps - 用
requestAnimationFrame驱动 SVGtransform却没开启 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代码职责分离与架构建议


urrentTime - lastTime;
lastTime = currentTime;