javascript中如何实现动画效果_如何使用requestAnimationFrame?

发布时间 - 2025-12-29 00:00:00    点击率:
requestAnimationFrame 是实现流畅动画的最优选择,它与屏幕刷新率同步、节省资源且避免丢帧;应配合 transform 和 will-change 提升性能,避免直接修改 left/top 触发布局。

在 JavaScript 中实现流畅动画,requestAnimationFrame 是比 setTimeoutsetInterval 更优的选择——它由浏览器统一调度,与屏幕刷新率同步(通常 60fps),节省资源且避免丢帧。

requestAnimationFrame 的基本用法

它接收一个回调函数,该函数会在下一次浏览器重绘前执行。和 setTimeout 不同,它不接受毫秒参数,而是“交给浏览器决定何时执行”:

示例:让一个 div 水平移动

const box = document.getElementById('box');
let left = 0;

function animate() {
  left += 2;
  box.style.left = left + 'px';
  
  // 继续下一帧,直到停止条件满足
  if (left < 400) {
    requestAnimationFrame(animate);
  }
}

requestAnimationFrame(animate); // 启动动画

为什么不能直接用 setInterval?

setInterval(animate, 16) 看似模拟 60fps,但存在几个问题:

  • 定时器触发时间不精确,可能与屏幕刷新不同步,导致卡顿或跳帧
  • 页面切到后台时,setInterval 仍运行(部分浏览器会降频,但不可靠),浪费 CPU
  • requestAnimationFrame 在标签页不可见时自动暂停,恢复时继续,更省电、更友好

封装可控制的动画函数

实际开发中,常需启动、暂停、取消动画。可以封装一个基础控制器:

class AnimationFrame {
  constructor(callback) {
    this.callback = callback;
    this.id = null;
  }

  start() {
    if (this.id !== null) return;
    const loop = () => {
      this.callback();
      this.id = requestAnimationFrame(loop);
    };
    this.id = requestAnimationFrame(loop);
  }

  stop() {
    if (this.id !== null) {
      cancelAnimationFrame(this.id);
      this.id = null;
    }
  }
}

// 使用
const anim = new AnimationFrame(() => {
  // 动画逻辑,比如更新 transform、opacity 等
});
anim.start();
// anim.stop();

配合 CSS transform 和 will-change 提升性能

纯 JS 修改 left/top 触发布局(layout)和绘制(paint),开销大。推荐:

  • transform: translateX() 替代 left —— 只触发合成(composite),GPU 加速
  • 对频繁动画元素加 style.willChange = 'transform'(谨慎使用,仅对确实需要的元素)
  • 优先使用 opacity 动画,也是合成层友好属性


# css  # javascript  # java  # js  # 浏览器  # 回调函数  # ai  # 重绘  # 为什么 


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


相关推荐: 如何生成腾讯云建站专用兑换码?  Laravel怎么创建控制器Controller_Laravel路由绑定与控制器逻辑编写【指南】  如何在Windows 2008云服务器安全搭建网站?  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  JS中页面与页面之间超链接跳转中文乱码问题的解决办法  Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】  如何选择PHP开源工具快速搭建网站?  网站制作大概要多少钱一个,做一个平台网站大概多少钱?  JS经典正则表达式笔试题汇总  Laravel如何使用Livewire构建动态组件?(入门代码)  Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】  如何用AI一键生成爆款短视频文案?小红书AI文案写作指令【教程】  英语简历制作免费网站推荐,如何将简历翻译成英文?  Laravel项目结构怎么组织_大型Laravel应用的最佳目录结构实践  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  如何在搬瓦工VPS快速搭建网站?  实例解析angularjs的filter过滤器  利用 Google AI 进行 YouTube 视频 SEO 描述优化  Laravel怎么创建自己的包(Package)_Laravel扩展包开发入门到发布  如何在Ubuntu系统下快速搭建WordPress个人网站?  用v-html解决Vue.js渲染中html标签不被解析的问题  如何快速搭建支持数据库操作的智能建站平台?  如何在阿里云服务器自主搭建网站?  Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践  制作公司内部网站有哪些,内网如何建网站?  Laravel如何配置.env文件管理环境变量_Laravel环境变量使用与安全管理  如何确认建站备案号应放置的具体位置?  javascript读取文本节点方法小结  装修招标网站设计制作流程,装修招标流程?  齐河建站公司:营销型网站建设与SEO优化双核驱动策略  如何快速上传建站程序避免常见错误?  黑客入侵网站服务器的常见手法有哪些?  Laravel如何发送邮件和通知_Laravel邮件与通知系统发送步骤  Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】  如何在万网自助建站平台快速创建网站?  Laravel如何实现数据库事务?(DB Facade示例)  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  如何实现javascript表单验证_正则表达式有哪些实用技巧  Python企业级消息系统教程_KafkaRabbitMQ高并发应用  今日头条微视频如何找选题 今日头条微视频找选题技巧【指南】  Java遍历集合的三种方式  jQuery validate插件功能与用法详解  大同网页,大同瑞慈医院官网?  如何快速搭建高效WAP手机网站吸引移动用户?  php打包exe后无法访问网络共享_共享权限设置方法【教程】  如何用wdcp快速搭建高效网站?  Firefox Developer Edition开发者版本入口  Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南