深入解读Promise对象
promise对象初印象:
promise对象是异步编程的一种解决方案,传统的方法有回调函数和事件,promise对象是一个容器,保存着未来才会结束的事件的结果
promise对象有两个特点:
promise新建后就会立即执行:
let promise = new Promise(function (resolve, reject) {
console.log("promise");
resolve("当前脚本所有同步任务执行完才会执行"); // resolve()将状态改为resolve,只有执行resolve()才会继续往下走,也就是调用then()
});
promise.then(function (value) {
console.log(value);
});
console.log("顺序执行");
// promise
// 顺序执行
// 当前脚本所有同步任务执行完才会执行
异步加载图:
function loadImageAsync(url) {
return new Promise((resolve, reject) => {
var img = new Image();
img.onload = function(){
resolve(img);
}
img.onerror = function(){
reject(new Error("Could not load image at"+url));
}
img.src = url;
});
}
loadImageAsync("./2.png").then((value) => {
document.body.appendChild(value);
value.style.border = "solid 5px red";
});
使用promise对象实现ajax操作:
function getJson(url) {
var promise = new Promise((resolve, reject) => {
var xhr = new XMLHttpRequest();
xhr.open("get", url, true);
xhr.onreadystatechange = function () {
if (xhr.readyState !== 4) {
return;
}
if (this.status === 200) {
resolve(this.response);
}else{
reject(new Error(this.statusText));
}
}
xhr.send();
});
return promise;
}
getJson("a.txt").then((json) => {
console.log(json);
},(error) => {
console.log("出错了");
});
链式操作用法:
function runAsync1(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步任务1执行完成");
resolve("数据1");
},2000);
});
}
function runAsync2(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步任务2执行完成");
resolve("数据2");
},2000);
});
}
function runAsync3(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步任务3执行完成");
resolve("数据3");
},2000);
});
}
runAsync1().then((data) => {
console.log(data);
return runAsync2();
}).then((data) => {3
console.log(data);
return runAsync3();
}).then((data) => {
console.log(data);
});
reject用法:
function getNumber(){
return new Promise((resolve,reject) => {
// 做一些异步操作
setTimeout(function(){
var num = Math.floor(Math.random() * 10);
if(num < 5){
resolve(num);
}else{
reject("数字太大了");
}
},2000);
});
}
getNumber().then((value) => {
console.log("resolve");
console.log(value);
},(error) => {
console.log("reject");
console.log(error);
})
catch的用法1:代替then()中的第二个参数
function getNumber() {
return new Promise((resolve, reject) => {
setTimeout(function () {
var num = Math.floor(Math.random() * 10);
if (num < 5) {
resolve(num);
} else {
reject("数字太大了");
}
});
});
}
getNumber().then((value) => {
console.log("resolve");
console.log(value);
}).catch((error) => {
console.log("reject");
console.log(error);
})
catch的用法2:then()中的第一个参数中抛出异常,那么并不会报错卡死js,而跳到catch()中
function getNumber() {
return new Promise((resolve, reject) => {
setTimeout(function () {
var num = Math.floor(Math.random() * 10);
if (num < 5) {
resolve(num);
} else {
reject("数字太大了");
}
});
});
}
getNumber().then((value) => {
console.log("resolve");
console.log(value);
console.log(someData);
}).catch((error) => {
console.log("reject");
console.log(error);
})
all的用法:
function runAsync1(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步操作1完成");
resolve("data1");
},2000);
});
}
function runAsync2(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步操作2完成");
resolve("data2");
},1000);
});
}
function runAsync3(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步操作3完成");
resolve("data3");
},3000);
});
}
Promise.all([runAsync1(),runAsync2(),runAsync3()])
.then((data) => {
console.log(data);
});
race的用法:
function runAsync1(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步操作1完成");
resolve("data1");
},1000);
});
}
function runAsync2(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步操作2完成");
resolve("data2");
},2000);
});
}
function runAsync3(){
return new Promise((resolve,reject) => {
setTimeout(function(){
console.log("异步操作3完成");
resolve("data3");
},3000);
});
}
Promise.race([runAsync1(),runAsync2(),runAsync3()])
.then((data) => {
console.log(data);
});
2.用race给某个异步请求设置超时时间,并且在超时后执行相应的操作:
function getImg(url){
return new Promise((resolve,reject) => {
var img = new Image();
img.onload = function(){
resolve(img);
}
img.src = "2.png";
});
}
function delayTime(){
return new Promise((resolve,reject) => {
setTimeout(function(){
reject("图片请求超时了");
},5000);
});
}
Promise.race([getImg(),delayTime()])
.then((data) => {
document.body.appendChild(data);
}).catch((error) => {
console.log(error);
});
理解Promise.resolve():
Promise.resolve("data1").then((value) => {
console.log(value);
});
理解Promise.reject():
Promise.reject(new Error("出错了")).then((value) => {
console.log(value);
}).catch((error) => {
console.log(error);
});
每次调用then()和catch()都会返回一个新创建的promise对象:
案例一:
var promise1 = new Promise((resolve,reject) => {
var num = Math.floor(Math.random() * 10);
if(num < 5){
resolve(num);
}else{
reject("数字太大了");
}
});
var promise2 = promise1.then((value) => {
console.log(value);
});
var promise3 = promise2.catch((error) => {
console.log(error);
});
console.log(promise1 !== promise2); // true
console.log(promise1 !== promise3); // true
console.log(promise2 !== promise3); // true
案例二:
var promise1 = new Promise((resolve,reject) => {
resolve(1);
});
promise1.then((value) => {
return value * 2;
});
promise1.then((value) => {
return value * 2;
});
promise1.then((value) => {
console.log("1" + value); //
});
当一个promise对象变为成功状态时后面的promise对象并没有停运:
function runAsync1() {
return new Promise((resolve, reject) => {
setTimeout(function () {
console.log(1);
resolve(2);
}, 500);
});
}
function runAsync2() {
return new Promise((resolve, reject) => {
setTimeout(function () {
console.log(3);
resolve(4);
}, 1000);
});
}
Promise.race([runAsync1(), runAsync2()]).then((value) => {
console.log(value);
}).catch((error) => {
console.log(error);
});
//
//
//
注:
Promise.prototype.then() -> 返回Promise对象
Promise.prototype.catch() -> 返回Promise对象
Promise.prototype.finally() -> 返回Promise对象
Promise.all() -> 返回Promise实例
Promise.race() -> 返回Promise实例
Promise.resolve() -> 返回Promise对象
Promise.reject() -> 返回Promise对象
深入解读Promise对象的更多相关文章
- Promise对象解读
首先强调的是"Promise"是对象,也就是说与其他JavaScript对象的用法,没有什么两样:其次,它起到代理作用(proxy),充当异步操作与回调函数之间的中介.它使得异步操 ...
- angular学习笔记(二十八-附2)-$http,$resource中的promise对象
下面这种promise的用法,我从第一篇$http笔记到$resource笔记中,一直都有用到: HttpREST.factory('cardResource',function($resource) ...
- ES6深入学习记录(二)promise对象相关
1.Promise的含义 Promise是异步编程的一种解决方案,比传统的解决方案--回调函数和事件更合理和强大.ES6将其写进了语言标准,统一了用法,原生提供了promise对象. 所谓Promis ...
- es6中的promise对象
Promise是异步里面的一种解决方案,解决了回调嵌套的问题,es6将其进行了语言标准,同意了用法,提供了`promise`对象, promise对象有三种状态:pending(进行中) .Resol ...
- ES6的promise对象应该这样用
ES6修补了一位Js修真者诸多的遗憾. 曾几何时,我这个小白从js非阻塞特性的坑中爬出来,当我经历了一些回调丑陋的写法和优化的尝试之后,我深深觉得js对于多线程阻塞式的开发语言而言,可能有着其太明显的 ...
- Angularjs promise对象解析
1.先来看一段Demo,看完这个demo你可以思考下如果使用$.ajax如何处理同样的逻辑,使用ng的promise有何优势? var ngApp=angular.module('ngApp',[]) ...
- JavaScript异步编程(1)- ECMAScript 6的Promise对象
JavaScript的Callback机制深入人心.而ECMAScript的世界同样充斥的各种异步操作(异步IO.setTimeout等).异步和Callback的搭载很容易就衍生"回调金字 ...
- Promise对象
1.Promise思想:每一个异步任务立刻返回一个Promise对象,由于是立刻返回,所以可以采用同步操作的流程.这个Promises对象有一个then方法,允许指定回调函数,在异步任务完成后调用. ...
- angularJS中的Promise对象($q)的深入理解
原文链接:a better way to learn AngularJS - promises AngularJS通过内置的$q服务提供Promise编程模式.通过将异步函数注册到promise对象, ...
随机推荐
- [MCM] 多目标优化 MOP(multi-objective programming)
生活中许多问题都是由相互冲突和影响的多个目标组成.人们会经常遇到使多个目标在给定区域同时尽可能最佳的优化问题,也就是多目标优化问题.优化问题存在的优化目标超过一个并需要同时处理就成为多目标优化问题. ...
- redis在.net架构中的应用(1)--使用servicestack连接redis
引言:作为少有的.net架构下的大型网站,stackoverflow曾发表了一篇文章,介绍了其技术体系,原文链接http://highscalability.com/blog/2011/3/3/sta ...
- ansible的模块使用说明
参考官方链接: https://docs.ansible.com/ansible/latest/user_guide/intro_adhoc.html#parallelism-and-shell-co ...
- 【转】Android多进程总结一:生成多进程(android:process属性)
前言 正常情况下,一个apk启动后只会运行在一个进程中,其进程名为apk的包名,所有的组件都会在这个进程中运行,以下为DDMS的进程截屏: com.biyou.multiprocess为进程名,也是a ...
- 004_centos安装pip的几种方式及pip源
一. (1) yum -y install epel-release yum install python-pip pip install --upgrade pip (2) python脚本的一键安 ...
- QT插件+ROS 1 安装配置
测试环境: 系统版本:Ubuntu14.04 ROS版本:indigo QT版本:5.8.0 QtCreator安装 1 安装前准备,安装相应的GNU开发工具集和OpenGL开发库, 请注意安装软件都 ...
- Vim 去除因为 Unix 和 Windows 换行符不同带来的 ^M 问题
由于各操作系统对换行符的处理不同, Unix: \n Windows : \r\n Mac : \r 所以有时 Vim 打开的文件会有如下情况: 解决方法为:在 Vim 中执行命令 :%s/\r//g ...
- element not interactable,这种提示表示元素当前在页面上不可见
1.出现element not interactable,发现这个元素在页面上不可见,需要拖动下拉框才能看到这个元素 2.这个时候需要让元素在页面上可见,才可操作
- C# Oracle 时间字符串转时间类型
C# 字符串转时间类型 yyyy-MM-dd HH:mm:ss yyyy-MM-dd hh:mm:ss d 月中的某一天.一位数的日期没有前导零. dd 月中的某一天.一位数的日期有一个前导零. d ...
- CF833D Red-Black Cobweb 点分治、树状数组
传送门 统计所有路径的边权乘积的乘积,不难想到点分治求解. 边权颜色比例在\([\frac{1}{2},2]\)之间,等价于\(2B \geq R , 2R \geq B\)(\(R,B\)表示红色和 ...