Promise对象的基础入门学习

今天来学习下Promise吧!其实这在笔试上也是一个考点。

基本介绍

Promise对象是CommonJS(熟悉的名字吧- -)工作组提出的规范.Promise原本只是社区提出的构想,一些外部函数库率先实现了该功能,ES6中将其写入了语言标准.

目的:为异步操作提供统一接口

Promise是啥,它就是一个javascript中一个对象,起着代理作用,充当异步操作与回调函数之间的中介。

避免类似于

这种嵌套地狱的产生.让我们的代码变得更加简单易读使用了Promise,大家都说好


  1. (new Promise(f1).then(f2)); 

总结:Promise使得异步操作的向下发展变成横向发展,程序流程变得清晰,易于阅读。

基本思想

  • 异步任务返回一个Promise对象,它有三种状态

1.pending(未完成)

2.resolved,fulfilled(已完成)

3.rejected(失败)

  • 它有两种变化途径

1.pending --> resolved/fulfilled

2.pending --> rejected

  • 它有两种结果

1.异步操作成功,返回一个值,状态变为resolved

2.异步操作失败,抛出一个错误,状态变为rejected

Promise使用.then()方法添加回调函数,then接收两个回调函数,第一个为成功时的回调函数,另一个为失败时的回调函数.主要为状态改变时调用相对的回调函数.

而且then可以链式调用。

基本使用

Promise构造函数接受一个函数作为参数,而该函数两个参数分别是resolve和reject.它们由JS引擎提供,不需要自己部署.


  1. Promise(function(resolve,reject){}) 

resolve函数作用为:将Promise对象从未完成变为成功(Pending->Resolved),异步操作成功时调用,并将异步操作的结果作为参数传递出去.

reject函数作用为:将Promise对象从未完成变为失败(Pending->Rejected),异步操作失败时调用,并将异步操作报出的错误作为参数传递出去.

Promise.then()方法可以用于指定Resolved状态和Reject状态的回调函数.


  1. promise.then(function(value){//成功+_+!},function(value){//失败Q_Q}); 

我们只想对异常进行处理时可以采用promise.then(undefined, onRejected)这种方式,或者promise.catch(onRejected)

!注意!此处有坑,接下来在深入节会进行讲解

Promise.all()方法接收一个promise对象的数组为参数,当这个数组中所有的Promise对象全部变成resolve/reject状态的时候,才会调用.then方法,其中传入的promise是同时开始,并行执行的。


  1. promise.all([promise1,promise2,.....]); 

Promise.race()方法和Promise.all()方法一样接收一个promise对象的数组作为参数,但是数组中有一个promise对象进入fulfilled或rejected状态,就会开始后续处理.


  1. promise.race([promise1,promise2,.....]); 

相关的语法糖


  1. Promise.resolve(42); 
  2. //等价于 
  3. new Promise(function(resolve){ 
  4.     resolve(42); 
  5. }); 
  6.  
  7. Promise.reject(new Error("出错了")); 
  8. //等价于 
  9. new Promise(function(resolve,reject){ 
  10.     reject(new Error("出错了")); 
  11. });  

深入

关于Thenable对象

这是非常类似于Promise的东西,拥有.then方法.

其中比较经典的例子就是jQuery.ajax()返回的值就是thenable的.


  1. var promise = Promise.resolve($.ajax('/json/comment.json')); 

这样就可以将thenable对象转化为promise对象

传送门:Promise.resolve()

关于promise设计:总是异步操作

看代码就能明白这个地方的问题了.


  1. var promise = new Promise(function (resolve){ 
  2.     console.log("inner promise"); // 1 
  3.     resolve(42); 
  4. }); 
  5. promise.then(function(value){ 
  6.     console.log(value); // 3 
  7. }); 
  8. console.log("outer promise"); // 2 
  9. //结果是 
  10. /* 
  11. inner promise // 1 
  12. outer promise // 2 
  13. 42            // 3 
  14. */  

可以看出,即使我们调用promise.then时promise对象已经确定状态,Promise也会以异步的方式调用回调函数,这就是Promise设计上的规定方针.

关于调用then/catch

每次调用then/catch,都会返回一个promise对象,这一点上我们通过使用===就可以判断出来每次promise对象其实都是不一样的

then和catch的错误处理区别

这点和上一点联合起来很容易理解

直接上图吧,来自于JavaScript Promise迷你书(中文版)

在结合我们的代码吧


  1. // <1> onRejected不会被调用 
  2. function badMain(onRejected) { 
  3.     return Promise.resolve(42).then(throwError, onRejected); 
  4. // <2> 有异常发生时onRejected会被调用 
  5. function goodMain(onRejected) { 
  6.     return Promise.resolve(42).then(throwError).catch(onRejected); 
  7. }  

onFullfilled中发生的错误,如在<1>里面throwError中的错误,是不会导致onRejected的执行(捕获异常)的,我们只能通过后面的catch方法才能捕获.

基本应用

不兼容方面

  1. 不兼容就是用polyfill吧
  2. 关于IE8以及以下版本中,catch会由于在ES3中为保留字,导致identifier not
    found错误,对此我们可以通过["catch"]或者then(undefined,function(){})来进行catch,而某些类库中,采用了caught作为函数名来规避该问题.值得注意的是,有很多压缩工具中自带了.catch转["catch"]

应用示例:

加载图片


  1. var preloadImage = function(path){ 
  2.   return new Promise(function(resolve,reject){ 
  3.     var image = new Image(); 
  4.     image.onload = resolve; 
  5.     image.onerror = reject; 
  6.     image.src = path; 
  7.   }) 
  8. preloadImage("https://dn-anything-about-doc.qbox.me/teacher/QianDuan.png").then(function(){ 
  9.   alert("图片加载成功"); 
  10. },function(){ 
  11.   alert("图片加载失败"); 
  12. })  

Ajax操作


  1. function search(term) { 
  2.     var url = 'http://example.com/search?q=' + term; 
  3.     var xhr = new XMLHttpRequest(); 
  4.     var result; 
  5.     var p = new Promise(function(resolve, reject) { 
  6.         xhr.open('GET', url, true); 
  7.         xhr.onload = function(e) { 
  8.             if (this.status === 200) { 
  9.                 result = JSON.parse(this.responseText); 
  10.                 resolve(result); 
  11.             } 
  12.         }; 
  13.         xhr.onerror = function(e) { 
  14.             reject(e); 
  15.         }; 
  16.         xhr.send(); 
  17.     }); 
  18.     return p; 
  19. search("Hello World").then(console.log, console.error);  

回到最初吧,其实Promise对象优点还是在于规范的链式调用,可以清晰看出程序流程.并且对于错误还能定义统一的处理方法。

作者:thewindsword

来源:51CTO

时间: 2025-01-20 17:53:14

Promise对象的基础入门学习的相关文章

移动交互小白零基础入门学习笔记之:手势篇

此贴为零基础入门学习贴,总结和积累些基础知识~ 1.基本手势 触屏设备中多样的手势操作,都是由这10种基本手势组合演变而来. 2.常用动作 基本动作是触屏界面中最常用的动作,如打开.选择等. 与对象有关的动作是对屏幕上某一目标对象的操作,如调整图片的位置大小,选择.删除或移动一个文件等. 导览动作是对屏幕视图的操作,如切换屏幕.滚动屏幕.缩放网页等. 画图示意动作是用画图的方式来示意某些操作,这些图形最好是常见易画的图形,符合用户心理预期,如画勾表示确认,画叉表示取消. 以上都是触屏设备中常用的

整理Javascript基础入门学习笔记_javascript技巧

了解什么是变量? 变量是用于存储信息的容器 变量的声明 语法: var  变量名变量名 = 值; 变量要先声明再赋值 变量可以重复赋值变量的命名规则 变量必须以字母开头: 变量也能以$和_符号开头(不过我们不推荐这么做): 变量名称对大小写敏感(a和A是不同的变量). 1.语句 语句以一个分号结尾:如果省略分号,则由解析器确定语句的结尾. 有个好的编码习惯,都要以 ; 结尾 2.数据类型 在JavaScript中,一段信息就是一个值(value).值有不同的类型,大家最熟悉的类型是数字.字符串(

jsp Servlet基础入门学习:访问CGI变量

cgi|js|servlet|变量|访问     6.1 CGI变量概述 如果你是从传统的CGI编程转而学习Java Servlet,或许已经习惯了"CGI变量"这一概念.CGI变量汇集了各种有关请求的信息: 部分来自HTTP请求命令和请求头,例如Content-Length头: 部分来自Socket本身,例如主机的名字和IP地址: 也有部分与服务器安装配置有关,例如URL到实际路径的映射. 6.2 标准CGI变量的Servlet等价表示 下表假定request对象是提供给doGet和

Jsp Servlet基础入门学习篇处理Cookie

9.1 Cookie概述 Cookie是服务器发送给浏览器的体积很小的纯文本信息,用户以后访问同一个Web服务器时浏览器会把它们原样发送给服务器.通过让服务器读取它原先保存到客户端的信息,网站能够为浏览者提供一系列的方便,例如在线交易过程中标识用户身份.安全要求不高的场合避免用户重复输入名字和密码.门户网站的主页定制.有针对性地投放广告,等等. Cookie的目的就是为用户带来方便,为网站带来增值.虽然有着许多误传,事实上Cookie并不会造成严重的安全威胁.Cookie永远不会以任何方式执行,

jsp Servlet基础入门学习:读取HTTP请求头

js|servlet|请求     5.1 HTTP请求头概述 HTTP客户程序(例如浏览器),向服务器发送请求的时候必须指明请求类型(一般是GET或者POST).如有必要,客户程序还可以选择发送其他的请求头.大多数请求头并不是必需的,但Content-Length除外.对于POST请求来说Content-Length必须出现. 下面是一些最常见的请求头: Accept:浏览器可接受的MIME类型. Accept-Charset:浏览器可接受的字符集. Accept-Encoding:浏览器能够

jsp Servlet基础入门学习:处理Cookie

cookie|js|servlet      9.1 Cookie概述 Cookie是服务器发送给浏览器的体积很小的纯文本信息,用户以后访问同一个Web服务器时浏览器会把它们原样发送给服务器.通过让服务器读取它原先保存到客户端的信息,网站能够为浏览者提供一系列的方便,例如在线交易过程中标识用户身份.安全要求不高的场合避免用户重复输入名字和密码.门户网站的主页定制.有针对性地投放广告,等等. Cookie的目的就是为用户带来方便,为网站带来增值.虽然有着许多误传,事实上Cookie并不会造成严重的

jsp Servlet基础入门学习:会话状态

js|servlet     10.1 会话状态概述 HTTP协议的"无状态"(Stateless)特点带来了一系列的问题.特别是通过在线商店购物时,服务器不能顺利地记住以前的事务就成了严重的问题.它使得"购物篮"之类的应用很难实现:当我们把商品加入购物篮时,服务器如何才能知道篮子里原先有些什么?即使服务器保存了上下文信息,我们仍旧会在电子商务应用中遇到问题.例如,当用户从选择商品的页面(由普通的服务器提供)转到输入信用卡号和送达地址的页面(由支持SSL的安全服务器

jsp Servlet基础入门学习:JSP动作

js|servlet      JSP动作利用XML语法格式的标记来控制Servlet引擎的行为.利用JSP动作可以动态地插入文件.重用JavaBean组件.把用户重定向到另外的页面.为Java插件生成HTML代码. JSP动作包括: jsp:include:在页面被请求的时候引入一个文件. jsp:useBean:寻找或者实例化一个JavaBean. jsp:setProperty:设置JavaBean的属性. jsp:getProperty:输出某个JavaBean的属性. jsp:forw

jsp Servlet基础入门学习:第一个Servlet

js|servlet      3.1 Servlet基本结构 下面的代码显示了一个简单Servlet的基本结构.该Servlet处理的是GET请求,所谓的GET请求,如果你不熟悉HTTP,可以把它看成是当用户在浏览器地址栏输入URL.点击Web页面中的链接.提交没有指定METHOD的表单时浏览器所发出的请求.Servlet也可以很方便地处理POST请求.POST请求是提交那些指定了METHOD="POST"的表单时所发出的请求,具体请参见稍后几节的讨论. import java.io