promise结合requestAnimationFrame用法示例

目录
  • promise基础用法
  • promise结合requestAnimationFrame
    • 理解requestAnimationFrame
    • 结合promise与requestAnimationFrame

promise基础用法

js为了解决单线程的异步执行问题,引入了事件循环队列体系,这个体系里的队列中都是一个个排着队等待执行的宏任务,settimeout就是一个宏任务,promise是典型的微任务,微任务概念是相对宏任务而言的,可以把微任务理解为宏任务中的队列。因为宏任务是按序执行,所以如果当前宏任务有微任务,只有等当前宏任务的所有微任务执行完毕,才会执行下一个宏任务。所以promise与settimeout不分前后紧挨着出现在代码里,那一定是先执行完promise的回调才会去执行settimeout。这里有一个例子:

setTimeout(()=>console.log("d"), 0)
var r = new Promise(function(resolve, reject){
     resolve()
});
r.then(() => {
		var begin = Date.now();
		while(Date.now() - begin < 1000);
		console.log("c1")
		new Promise(function(resolve, reject){
        resolve()
		}).then(() => console.log("c2"))
});
// c1 c2 d

输出顺序是c1、c2、d,settimeout后面紧接着一个promise,那得先执行promise,在promise的then回调中,先强行等了一秒钟,输出c1后又执行了一个微任务;这一流程走完了,最后才回头执行了settimeout。这个例子能很好地说明promise(微任务)与settimeout(宏任务)的关系。

promise结合requestAnimationFrame

promise在项目中会有各种应用场景,但是promise结合requestAnimationFrame的用法比较少见,我在项目中正好遇到这种需求了。

先介绍下requestAnimationFrame,它常常用来写动画,相比setInterval、 settimeout这些更加精确而且性能更好。前面在介绍promise的时候说到宏任务与微任务的概念,requestAnimationFrame也是一种宏任务。

现在的需求就是要在一个动画结束之后再调用另一个函数,这个过程可以简单形象地理解为“图穷匕首见”。

理解requestAnimationFrame

先从mdn 摘个动画的代码,理解下requestAnimationFrame的用法:

<div id='alimate' style="background-color: brown; width: 100px;height: 100px;"></div>
<script>
const element = document.getElementById('alimate');
let start;
function step(timestamp) {
  if (start === undefined)
    start = timestamp;
  const elapsed = timestamp - start;
  //这里使用`Math.min()`确保元素刚好停在200px的位置。
  element.style.transform = 'translateX(' + Math.min(0.1 * elapsed, 200) + 'px)';
  if (elapsed < 2000) { // 在两秒后停止动画
    window.requestAnimationFrame(step);
  }
}
window.requestAnimationFrame(step);
</script>

我在这上面加了点代码,可以直接看到效果。这个官方例子非常值得研究,这是我摘抄的原因,有几点要说明下:

  • step函数的参数timestamp是有值的,它是一个double类型的十进制数,表示当前回调函数step被调用的时间
  • 两秒后取消动画,动画之所以停止,只不过是两秒后,没有再触发回调step了
  • 要想取消回调,可以用cancelAnimationFrame,它的参数是requestAnimationFrame的返回值

后面有时间可以研究下requestAnimationFrame的垃圾回收,如果在上面的代码中不加条件限制,这个持续动画对性能的影响是个值得研究的问题。

结合promise与requestAnimationFrame

结合上文来看,requestAnimationFrame是一个宏任务,那在promise中写一个宏任务,这事我们经常做,就像这样:

var f = new Promise((resolve, reject) => {
		console.log('a')
		setTimeout(resolve,100);
})
f.then(() => { console.log('res') })

那套用到requestAnimationFrame也是一样的,结合上面的用法示例,我们可以这样写:

<div id='alimate' style="background-color: brown; width: 100px;height: 100px;"></div>
<script>
const element = document.getElementById('alimate');
let start;
var f = new Promise((resolve, reject) => {
		function step(timestamp) {
		  if (start === undefined)
		    start = timestamp;
		  const elapsed = timestamp - start;
		  //这里使用`Math.min()`确保元素刚好停在200px的位置。
		  element.style.transform = 'translateX(' + Math.min(0.1 * elapsed, 200) + 'px)';
		  if (elapsed < 2000) { // 在两秒后停止动画
		    requestAnimationFrame(step);
		  }else {
				resolve()
			}
		}
		requestAnimationFrame(step);
})
f.then(() => {
		element.innerHTML= "动画结束了!";
})
</script>

需要注意的是,这个promise里面的主体,其实是一个requestAnimationFrame函数,模型可以简单写成这样:

new Promise((resolve, reject) => {
		requestAnimationFrame(f)
}).then()

问题的关键就是回调函数f要写在哪里!?如果写在promise之外,在动画结束,需要触发promise的回调时,就有问题了,看代码:

new Promise((resolve, reject) => {
		requestAnimationFrame(f)
}).then()
function f () {
   // 动画结束之后,如何触发resolve
}

据我了解的,resolve只能在promise内部调用,而Promise.resolve的用法,只适合去封装异步函数,在这种情况下,最好就是像我那样写,把requestAnimationFrame和它的回调函数都封装在promiset中,等待时机成熟就调用resolve就可以了。

如果非要写在外面也是有办法的,把resolve当作参数传出去,不过要注意的是resolve必须是第二个参数:

<div id='alimate' style="background-color: brown; width: 100px;height: 100px;"></div>
<script>
const element = document.getElementById('alimate');
let start;
function step(timestamp,resolve) {
		  if (start === undefined)
		    start = timestamp;
		  const elapsed = timestamp - start;
		  //这里使用`Math.min()`确保元素刚好停在200px的位置。
		  element.style.transform = 'translateX(' + Math.min(0.1 * elapsed, 200) + 'px)';
		  if (elapsed < 2000) { // 在两秒后停止动画
		    requestAnimationFrame(step);
		  }else {
				resolve()
			}
		}
var f = new Promise((resolve, reject) => {
		var s;
		requestAnimationFrame(step(s, resolve));
})
f.then(() => {
		element.innerHTML= "动画结束了!";
})

这样写的后果就是step的第一个参数失效了,不再是一个时间戳了,不过你可以自己写一个,动画也可以实现。再强调下resolve要先放在第二个,如果你把resolve写在第一个,那step得到的resolve就不再是resolve函数,而是一个时间戳,这样的话resolve就无法执行了!

参考资料:

developer.mozilla.org/zh-CN/docs/…

stackoverflow.com/questions/6…

以上就是promise结合requestAnimationFrame用法示例的详细内容,更多关于promise requestAnimationFrame的资料请关注我们其它相关文章!

(0)

相关推荐

  • 详解requestAnimationFrame和setInterval该如何选择

    目录 正文 requestAnimationFrame的工作方式 setInterval的工作方式 requestAnimationFrame和setInterval之间的区别 适用场景 写在后面 正文 在Web前端开发中,使用计时器是实现动画效果.周期性任务.定时器等常见操作的核心.JavaScript提供了两种计时器:requestAnimationFrame和setInterval.虽然它们看起来很相似,但它们的工作方式却有很大的不同. requestAnimationFrame的工作方式

  • JavaScript requestAnimationFrame动画详解

    进入web2.0时代,在网页中实现动画已经不再局限于一种方法 你可以用CSS3的animattion+keyframes; 你也可以用css3的transition; 你还可以用通过在canvas上作图来实现动画,也可以借助jQuery动画相关的API方便地实现; 当然最原始的你还可以使用window.setTimout()或者window.setInterval()通过不断更新元素的状态位置等来实现动画,前提是画面的更新频率要达到每秒60次才能让肉眼看到流畅的动画效果. 认识JavaScrip

  • JS使用window.requestAnimationFrame()实现逐帧动画

    window.requestAnimationFrame() 方法告诉浏览器您希望执行动画,并请求浏览器调用指定的函数在下一次重绘之前更新动画.该方法使用一个回调函数作为参数,这个回调函数会在浏览器重绘之前调用. 如果你想做逐帧动画的时候,你应该用这个方法.这就要求你的动画函数执行会先于浏览器重绘动作.通常来说,被调用的频率是每秒60次,但是一般会遵循W3C标准规定的频率.如果是后台标签页面,重绘频率则会大大降低. 基本语法: requestID = window.requestAnimatio

  • requestAnimationFrame使用示例详解

    目录 requestAnimationFrame--use是什么 回调函数 调用频率 返回啥 如何用window.requestAnimationFrame(callback); 示例 多个运行情况 停止执行 cancelAnimationFrame 动画运用 总结 requestAnimationFrame--use是什么 告诉浏览器用来执行一个动画,并且在下一次重绘之前调用其指定的回调函数取更新动画,所以该方法的参数就是一个回调函数,在下一次重绘时候调用. 回调函数 回调函数即是传入的cal

  • promise结合requestAnimationFrame用法示例

    目录 promise基础用法 promise结合requestAnimationFrame 理解requestAnimationFrame 结合promise与requestAnimationFrame promise基础用法 js为了解决单线程的异步执行问题,引入了事件循环队列体系,这个体系里的队列中都是一个个排着队等待执行的宏任务,settimeout就是一个宏任务,promise是典型的微任务,微任务概念是相对宏任务而言的,可以把微任务理解为宏任务中的队列.因为宏任务是按序执行,所以如果当

  • C语言指针详解及用法示例

    新手在C语言的学习过程中遇到的最头疼的知识点应该就是指针了,指针在C语言中有非常大的用处.下面我就带着问题来写下我对于指针的一些理解. 指针是什么? 指针本身是一个变量,它存储的是数据在内存中的地址而不是数据本身的值.它的定义如下: int a=10,*p; p=&a int a=10; int *p=&a; 首先我们可以理解 int* 这个是要定义一个指针p,然后因为这个指针存储的是地址所以要对a取地址(&)将值赋给指针p,也就是说这个指针p指向a. 很多新手都会对这两种定义方法

  • MySQL中or语句用法示例

    1.mysql中or语法的使用,在mysql语法中or使用注意点. 项目遇到坑,遍历发放奖励数据查询错误!!! $sql = 'SELECT * FROM `vvt_spread_doubleegg_exchange_award` AS p WHERE p.`act_type` = 4 or p.`act_type` = 5 AND p.`user_id` = ' .$user_id ; sql中的or语法一般用于多个条件的查询,上面的语法查询的相当于:两个sql查询出来的数据集合. $sql

  • Function.prototype.bind用法示例

    复制代码 代码如下: //ECMAScript 5 Function.prototype.bind函数兼容处理 (function(){ if ( !Function.prototype.bind ) { //function(){}.bind Function.prototype.bind = function ( o, /*参数列表*/ ) { var self = this, boundArgs = Array.prototype.slice.call(arguments, 0); ret

  • python中hashlib模块用法示例

    我们以前介绍过一篇Python加密的文章:Python 加密的实例详解.今天我们看看python中hashlib模块用法示例,具体如下. hashlib hashlib主要提供字符加密功能,将md5和sha模块整合到了一起,支持md5,sha1, sha224, sha256, sha384, sha512等算法 具体应用 #!/usr/bin/env python # -*- coding: UTF-8 -*- #pyversion:python3.5 #owner:fuzj import h

  • laravel学习笔记之模型事件的几种用法示例

    前言 本文主要给大家介绍了关于laravel模型事件用法的相关内容,文中通过示例代码介绍了laravel模型事件的多种用法,下面话不多说了,来一起看看详细的介绍吧. 用法示例 一 .简单粗鲁(用于本地测试) 路由中定义: Event::listen('eloquent.updated: App\Post',function (){ dump('测试一下修改事件'); }); Route::post('/post/{id}', 'PostController@update'); 二 .生成事件和监

  • Kryo序列化及反序列化用法示例

    Kryo 是一个快速高效的 Java 对象图形序列化框架,主要特点是性能.高效和易用.该项目用来序列化对象到文件.数据库或者网络. 代码地址:https://github.com/EsotericSoftware/kryo 样例代码地址:https://github.com/nettm/public Kryo的序列化及反序列速度很快,据说很多大公司都在用.我在把对象序列化都转换成了字符串形式,是为了把对象存储到缓存中.我们日常项目中使用的数据形式包括对象.List.Set和Map,因此主要把这几

  • bootstrap paginator分页前后台用法示例

    bootstrap paginator分页前后台用法示例,供大家参考,具体内容如下 准备工作: bootstrap-paginator.js 插件 github开源项目百度自行下载 引入js文件(JQuery1.8+bootstrap.min.js+bootstrap.css+bootstrap-paginator.js) <!--路径根据自己项目修改--> <link rel="stylesheet" href="/bootstrap/css/bootst

  • jQuery弹出层插件popShow(改进版)用法示例

    本文实例讲述了jQuery弹出层插件popShow(改进版)用法.分享给大家供大家参考,具体如下: 前面一篇<jQuery弹出层插件popShow用法示例>分析了popShow插件的基本用法,这里再对插件进行一番改进. popShow弹出层 图1.1 弹出层效果 1.引入JS和CSS文件 <link href="popShow.css" rel="stylesheet" type="text/css" /> <scr

  • PHP实现的简单分页类及用法示例

    本文实例讲述了PHP实现的简单分页类及用法.分享给大家供大家参考,具体如下: <?php /* * 使用: * $page = new Page(连接符,查询语句,当前页码,每页大小,页码符) * 连接符:一个MYSQL连接标识符,如果该参数留空,则使用最近一个连接 * 查询语句:SQL语句 * 当前页码:指定当前是第几页 * 每页大小:每页显示的记录数 * 页码符:指定当前页面URL格式 * * 使用例子: * $sql = "select * from aa"; * $pag

随机推荐