继续填坑

模式

  考虑下面的代码:

    function fn(x) {
//do something
return new Promise(function(resolve, reject) {
//调用resolve(..)和reject(...)
});
}
var p = fn(2);

  new Promise(..)模式通常称为revealing constructor。传入函数会立即执行(不会像then(..)中的回调一样异步延迟),它有两个参数,分别为resolve和reject。这些是promise的决议函数。resolve通常标识完成,reject标识拒绝。

鸭子类型

  如何判断一个对象是不是一个Promise?虽然promise是通过new Promise(..)创建的,但是无法通过instanceof Promise来检测,最主要的原因是promise可能来自其他的窗口,检查无法识别这样的Promise实例。

  识别Promise的方法被定义为是否具有then方法。也就是说,任何对象和函数,如果有then(包括原型链上)方法,就认为是一个Promise对象。

  逻辑大概如下:

    if (
p !== null &&
(typeof p === 'object' || typeof p === 'function') &&
typeof p.then === 'function'
) {
//这是一个promise对象
}

Promise特点

  1、对一个Promise调用then时,提供的回调永远会被异步调用。

  2、只要promise被决议,提供给then的回调会被自动调用,永远返回一个值。

promise与竞态

  虽然说一旦promise被决议,后续then方法的回调一定会被调用,但是决议本身未被执行,只能通过别的机制来强制执行:

    //必须返回一个promise
function delay(time) {
return new Promise(function(resolve, reject) {
setTimeout(function() {
reject('time out');
}, time);
});
}
//3秒内未决议就会被reject
Promise.race(p, delay(3000)).then(function() {
//success
}, function(err) {
//failed
})

  

Promise异常

  考虑下面的代码:

    var p = new Promise(function(resolve, reject) {
//异常
fo();
})
p.then(function(data) {
console.log(data);
}, function() {
//这个reject接受了异常 并又搞出了一个异常
foo();
}).then(function() { }, function() {
//这个reject处理了最后的异常
console.log(1);
});

  可以看出,每一个then方法会返回一个promise,并且一旦决议就不会改变,异常会转接到下一个then的reject。

Promise的回调

  看起来promise也是利用回调函数完成异步操作,但是有一些不同。

关于Promise.resolve(..)

  如果向Promise.resolve(..)传递一个非Promise、非thenable(即有then方法的对象或函数)的立即值,就会得到一个用这个值填空的promise,相当于包装。

  下面两种情况,p1和p2的行为是一样的:

    var p1 = new Promise(function(resolve, reject) {
resolve(42);
});
var p2 = Promise.resolve(42);
console.log(p2); //Promise {[[PromiseStatus]]: "resolved", [[PromiseValue]]: 42}

  而如果向Promise.resolve(..)传递一个真正的Promise,就只会返回同一个promise。

    var p1 = Promise.resolve(42);
var p2 = Promise.resolve(p1);
console.log(p1 === p2); //true

  如果向Promise.resolve(..)传递了一个非Promise的thenable值,前者会试图展开这个值,而且展开过程会持续到提取出一个具体的非类Promise的最终值。

  比如说:

    var p = {
then: function(resolve, reject) {
resolve(42);
// reject('123');
}
};
p.then(function resolve(val) {
console.log(val);
}, function reject(err) {
console.log(err);
});
Promise.resolve(p); //打印42 得到一个Promise

  Promise.resolve(..)可以接受任何thenable,将其解封为非thenable的值。从Promise.resolve(..)得到的肯定是一个Promise,是一个可信任的值。

  将一个未知工具封装Promise并解析,可以这样:

    Promise.resolve('tool').then(function(val) {
console.log(val);
});

  这样,可以将所有未知封装为Promise,保证异步调用。

链式调用

  可以把多个Promise连接到一起以表示一系列异步步骤:

  1、每次调用then(..),它都会创建并返回一个新的Promise,我们可以将其链接起来。

  2、不管从then(..)调用的完成回调返回的值是什么,都会被设置为被链接Promise。

  不太懂啊:

    var p = Promise.resolve(2);
var p2 = p.then(function(v) {
console.log(v); //
return v * 2;
});
p2.then(function(v) {
console.log(v); //
});

  通过Promise.resolve对2进行封装,调用第一个then方法得到数据2。接着返回一个包含4的Promise封装对象,由调用then方法获取到4。

  只要通过return和then方法,就能实现Promise链式调用。

Ajax案例

  链式调用可以与ajax完美结合,如下例:

    //假设存在方法ajax(url,callback)
function request(url) {
return new Promise(function(resolve, reject) {
//ajax的回调函数是promise的resolve()函数
ajax(url, resolve);
});
}
//异步请求返回数据data
request('http://...').then(function(data) {
//将返回的数据拼接到第二个url作为参数再次请求
return request('http://...?data=' + data);
}).then(function(data2) {
//得到最终数据
console.log(data2);
});

  如果then方法参数未提供resolve或者reject,会有一个默认的函数生成,如下例:

    Promise.resolve(2).then(
//function(v){
// return v;
//}
null,
function() {
console.log(1);
}).then(function(v) {
console.log(v); //
},
//默认错误处理函数
//function(err) {
// throw err;
//}
);

  

  小结:1、调用Promise.resolve(..)与then(..)总会返回一个Promise。

     2、决议后返回的值可以通过链式调用解决掉。

  

读书笔记-你不知道的JS中-promise(2)的更多相关文章

  1. 读书笔记-你不知道的JS中-promise

    之前的笔记没保存没掉了,好气,重新写! 填坑-- 现在与将来 在单个JS文件中,程序由许多块组成,这些块有的现在执行,有的将来执行,最常见的块单位是函数. 程序中'将来'执行的部分并不一定在'现在'运 ...

  2. 读书笔记-你不知道的JS中-promise(3)

    坑坑坑 关于术语:决议.完成以及拒绝. 首先观察Promise(..)构造器: var p = new Promise(function(x, y) { //x() 用于完成 //y() 用于拒绝 } ...

  3. 读书笔记-你不知道的JS中-函数生成器

    这个坑比较深 可能写完我也看不懂(逃 ES6提供了一个新的函数特性,名字叫Generator,一开始看到,第一反应是函数指针?然而并不是,只是一个新的语法. 入门 简单来说,用法如下: functio ...

  4. 读书笔记-你不知道的JS上-对象

    好想要对象··· 函数的调用位置不同会造成this绑定对象不同.但是对象到底是什么,为什么要绑定他们呢?(可以可以,我也不太懂) 语法 对象声明有两个形式: 1.字面量 => var obj = ...

  5. 读书笔记-你不知道的JS上-this

    关于this 与静态词法作用域不用,this的指向动态绑定,在函数执行期间才能确定.感觉有点像C++的多态? var a = 1; var obj = { a: 2, fn: function() { ...

  6. 读书笔记-你不知道的JS上-函数作用域与块作用域

    函数作用域 Javascript具有基于函数的作用域,每声明一个函数,都会产生一个对应的作用域. //全局作用域包含f1 function f1(a) { var b = 1; //f1作用域包含a, ...

  7. 读书笔记-你不知道的JS上-词法作用域

    JS引擎 编译与执行 Javascript引擎会在词法分析和代码生成阶段对运行性能进行优化,包含对冗余元素进行优化(例如对语句在不影响结果的情况下进行重新组合). 对于Javascript来说,大部分 ...

  8. 读书笔记-你不知道的JS上-混入与原型

    继承 mixin混合继承 function mixin(obj1, obj2) { for (var key in obj2) { //重复不复制 if (!(key in obj1)) { obj1 ...

  9. 读书笔记-你不知道的JS上-闭包与模块

    闭包定义 当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行. 看一段最简单的闭包代码: function foo() { var a = 2; //闭包 fun ...

随机推荐

  1. pig报错

    pig failed to read data from....... 错误可能1:load data的目录不在,或者引用出错,load data '/in/train'这里的红色/应该去掉,因为默认 ...

  2. 一篇搞定微信分享和line分享

    前言 在h5的页面开发中,分享是不可或缺的一部分,对于一些传播性比较强的页面,活动页之类的,分享功能极为重要.例如,京东等电商年末时会有一系列的总结h5在微信中传播,就不得不提到微信的分享机制. 微信 ...

  3. 入坑IT都快十年了

    一起帮的开发直播已经告一段落:一是主体的功能差不多都实现了,二是用到的架构技术都展示得差不多了.以后就算继续开发,也应该都是一些“技术上”重复的工作而已.整个直播过程耗时近半年,SVN提交1062次, ...

  4. JavaWeb(一)Servlet中乱码解决与转发和重定向的区别

    前言 前面其实已经把Servlet中所有的内容都介绍完了,这篇讲补充一点乱码和重定向与转发之间的区别! 一.request请求参数出现乱码问题 1.1.get请求 1)乱码示例 get请求的参数是在u ...

  5. Django进阶篇【2】

    在学习之前,我们补充一个知识点(static用法) 创建APP 配置: setting.py STATICFILES_DIRS = ( os.path.join(BASE_DIR,'static') ...

  6. java远程备份mysql数据库关键问题(限windows环境,亲测解决)

    其它环境同理也可解决. 条件:为了使用mysql命令,本机要安装mysql ,我本机安装的是mysql 5.5. 错误1:使用命令 mysqldump -h192.168.1.50 -u root - ...

  7. ch1-vuejs基础入门(hw v-bind v-if v-for v-on v-model 应用组件简介 小案例)

    1 hello world 引入vue.min.js 代码: ----2.0+版本 <div id="test"> {{str}} </div> <s ...

  8. marked插件在线实时解析markdown的web小工具

    访问地址: https://mdrush.herokuapp.com/ github项目: https://github.com/qcer/MDRush 实现简介: 1.动态数据绑定 借助Vuejs, ...

  9. 奥利奥好吃吗?Android 8.0新特性适配测试报告来啦!

    WeTest 导读 谷歌2017 I/O开发者大会上发布了Android 8.0的正式版, 其官方代号为Oreo(奥利奥).网上关于Android8.0新功能特性的介绍已铺天盖地,新功能特性会对程序应 ...

  10. Node Inspector 代理实现

    本文首发于 https://github.com/whxaxes/blog/issues/9 背景 平时做 node 开发的时候,通过 node inspector 来进行断点调试是一个很常用的 de ...