javascript动画如何实现_如何使用requestAnimationFrame创造流畅动画【教程】
发布时间 - 2026-02-02 00:00:00 点击率:次requestAnimationFrame 是实现流畅动画的唯一合理选择,因其与屏幕刷新率同步、后台自动暂停、无强制延迟抖动;需用时间戳计算 deltaTime 保证跨设备速度一致,避免混用 CSS 动画时的跳变。
requestAnimationFrame 是实现流畅 JavaScript 动画的唯一合理选择,setTimeout/setInterval 本质无法匹配屏幕刷新节奏,必然掉帧或卡顿。
为什么 requestAnimationFrame 比 setTimeout 更适合动画
浏览器在每次重绘前会主动调用 requestAnimationFrame 回调,它天然与显示器刷新率(通常是 60Hz)同步;而 setTimeout 只能“尽力”按毫秒间隔执行,实际触发时间受 JS 主线程阻塞、任务队列延迟等影响,容易累积误差。更关键的是:页面切到后台时,requestAnimationFrame 会被自动暂停,setTimeout 却照常运行,浪费资源甚至导致切回前台时动画突进。
- 不手动控制帧率 —— 浏览器决定何时执行,你只负责“下一帧该干什么”
- 自动节流 —— 隐藏标签页、系统节能模式下自动降频或暂停
- 无强制延迟抖动 —— 不像
setTimeout(fn, 16)实际可能延后 2–5ms
最简可用的 requestAnimationFrame 动画循环结构
核心是递归调用自身,并在每次回调中更新状态 + 渲染。别漏掉“停止条件”,否则无限执行:
let animationId = null;
const animate = () => {
// 更新逻辑:比如 this.x += this.speed
update();
// 渲染逻辑:比如 element.style.transform = `translateX(${this.x}px)`
render();
// 关键:下一次重绘前继续调用
animationId = requestAnimationFrame(animate);
};
// 启动
animationId = requestAnimationFrame(animate);
// 停止(例如离开页面或动画完成)
// cancelAnimationFrame(animationId);
- 必须把
requestAnimationFrame的返回值存起来,才能后续用cancelAnimationFrame中断 - 不要在回调里直接写
requestAnimationFrame(animate)而不赋值 —— 这会导致无法取消 - 更新(update)和渲染(render)应尽量轻量;重计算放 Web Worker,DOM 操作批量做
如何处理不同设备刷新率下的时间一致性
requestAnimationFrame 回调函数会接收一个高精度时间戳(DOMHigh

let lastTime = 0;
const animate = (currentTime) => {
const deltaTime = currentTime - lastTime;
lastTime = currentTime;
// 例如:每秒移动 100px,则这一帧应移动 (100 deltaTime / 1000) px
this.x += this.speed deltaTime / 1000;
render();
requestAnimationFrame(animate);
};
- 不用
Date.now()—— 它精度低,且不受浏览器节流策略保护 - 避免用固定增量(如
this.x += 2)—— 在 30Hz 屏幕上会慢一半 - 如果动画需严格帧数控制(如 24fps 过场),仍需用时间戳做采样过滤,而非硬等帧
常见踩坑:CSS 动画 vs JS 动画的边界在哪
不是所有动效都该用 requestAnimationFrame 手写。浏览器对 CSS transform/opacity 的动画做了硬件加速和合成层优化,性能远超 JS 操控 style;而 JS 适合做需要动态逻辑判断的动画(比如跟随鼠标、物理模拟、数据驱动形变)。混用时尤其注意:
- CSS 动画正在运行时,JS 直接改
element.style.transform会强行中断 CSS 动画并覆盖,造成跳变 - 用
getComputedStyle读取 transform 值再解析 —— 成本高,且返回的是 matrix,非原始值 - 想用 JS 控制但保留 CSS 加速?优先用
transform: translateZ(0)或will-change: transform提前升层,再用requestAnimationFrame改style.transform
真正难的不是写循环,而是判断该不该用它、怎么跟 CSS 协同、以及如何让时间计算不漂移 —— 这些细节没处理好,60fps 的承诺就只是幻觉。
# css
# javascript
# java
# js
# 浏览器
# 回调函数
# 显示器
# 硬件加速
# 重绘
# 为什么
# date
# 递归
# 循环
# 线程
# 主线程
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Mybatis 中的insertOrUpdate操作
Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能
网站制作报价单模板图片,小松挖机官方网站报价?
Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全
Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言
网站制作大概多少钱一个,做一个平台网站大概多少钱?
如何在自有机房高效搭建专业网站?
Laravel怎么导出Excel文件_Laravel Excel插件使用教程
标题:Vue + Vuex + JWT 身份认证的正确实践与常见误区解析
如何快速上传自定义模板至建站之星?
微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】
深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?
如何在万网ECS上快速搭建专属网站?
Laravel Fortify是什么,和Jetstream有什么关系
如何在云主机上快速搭建多站点网站?
HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】
高端建站如何打造兼具美学与转化的品牌官网?
如何快速登录WAP自助建站平台?
如何续费美橙建站之星域名及服务?
Python文件异常处理策略_健壮性说明【指导】
Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)
如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体
Laravel Livewire是什么_使用Laravel Livewire构建动态前端界面
浏览器如何快速切换搜索引擎_在地址栏使用不同搜索引擎【搜索】
Laravel怎么连接多个数据库_Laravel多数据库连接配置
Laravel如何处理异常和错误?(Handler示例)
JavaScript如何实现路由_前端路由原理是什么
Laravel Docker环境搭建教程_Laravel Sail使用指南
谷歌Google入口永久地址_Google搜索引擎官网首页永久入口
Win11任务栏卡死怎么办 Windows11任务栏无反应解决方法【教程】
ChatGPT怎么生成Excel公式_ChatGPT公式生成方法【指南】
Windows10如何更改计算机工作组_Win10系统属性修改Workgroup
想要更高端的建设网站,这些原则一定要坚持!
Laravel如何记录自定义日志?(Log频道配置)
,怎么在广州志愿者网站注册?
如何在局域网内绑定自建网站域名?
Laravel集合Collection怎么用_Laravel集合常用函数详解
Laravel如何实现用户注册和登录?(Auth脚手架指南)
专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?
JS弹性运动实现方法分析
深入理解Android中的xmlns:tools属性
如何用已有域名快速搭建网站?
如何在IIS7上新建站点并设置安全权限?
Laravel如何处理CORS跨域请求?(配置示例)
百度浏览器网页无法复制文字怎么办 百度浏览器复制修复
Laravel如何创建自定义Artisan命令?(代码示例)
详解Android图表 MPAndroidChart折线图
如何破解联通资金短缺导致的基站建设难题?
Laravel Blade组件怎么用_Laravel可复用视图组件的创建与使用
原生JS获取元素集合的子元素宽度实例

