在javascrpit的语言特性上 有明确的一个特性指出,该语言的是单线程进程。这就意味着JavaScript的所有网络操作,浏览器事件,都必须是异步执行。

如下面的例子,可以感受到单线程与异步回调:

function callback() {
console.log('Done');
}
console.log('before setTimeout()');
setTimeout(callback, 1000); // 1秒钟后调用callback函数
console.log('after setTimeout()');

将上面代码在控制台打出 我们可以看的到

before setTimeout()
after setTimeout()
(等待1秒后)
Done

可以看出,异步操作 会在未来的某个点 由回调函数触发

再来看一个典型的异步操作:ajax请求

// 这个是原生的ajax请求一部分  注册事件 onreadystatechange 状态改变就会调用
request.onreadystatechange = function () {
if (request.readyState === 4) { //判断服务器是否响应正确
if (request.status === 200) {
return success(request.responseText);
} else {
return fail(request.status);
}
}
}
// 另外 附上原生的ajax请求
//步骤一:创建异步对象
var ajax = new XMLHttpRequest();
//步骤二:设置请求的url参数,参数一是请求的类型,参数二是请求的url,可以带参数,动态的传递参数starName到服务端
ajax.open('get','getStar.php?starName='+name);
//步骤三:发送请求
ajax.send();
//步骤四:注册事件 onreadystatechange 状态改变就会调用
ajax.onreadystatechange = function () {
if (ajax.readyState==4 &&ajax.status==200) {
//步骤五 如果能够进到这个判断 说明 数据 完美的回来了,并且请求的页面是存在的
    console.log(ajax.responseText);//输入相应的内容
  }
}

对于上面的代码 有些繁琐,可以改写嘛?比如这样

var ajax = ajaxGet('http://...');
ajax.ifSuccess(success)
.ifFail(fail);

上面的代码 体现了一种链式写法,这种写法看起来比较简洁些,并且还体现了另外一个思想:

我不管你有没有请求成功,但是我发起了与服务器的通信,至于结果,我可以再在未来的某个代码片段里去调用 success 与 fail 的函数。

于是 这种 " 承诺于未来将会执行 " 的对象,被称为了promise  中文含义 : 承诺。

promise 有很多的开源实现 在ES6中被统一规范,由浏览器直接支持。可以测试你的浏览器是否支持promise。在控制台上打印以下代码:

new Promise(function () {
console.log('我支持promise')
})

既然知道了promise的大概含义,我们来实现一个这样的例子吧:下面代码里 我们默认当num<1时 是通信成功,不然就是通信失败。模拟一个ajax请求。

        function test(success,fail) {
var num = Math.random() * 2;
console.log('相应时间为'+ num +'秒')
setTimeout(function () {
if(num < 1) {
success('执行成功函数');
}else {
fail('执行失败的函数')
}
},num * 1000)
}

能够看出 , test()函数有两个参数,其实就是回调函数,当num < 1时 我们定义其为相应成功,调用success函数,失败时 就调用 fail 函数。由此可以看出:

test()函数只在关心自身的逻辑,并没有去关心 执行success 还是 fail 函数 如何去处理结果。

于是 我们可以用promise对象 来出来这个执行的函数。

        var p1 = new Promise(test);
var p2 = p1.then(function (data) {
console.log('成功' + data)
});
var p3 = p2.catch(function (data2) {
console.log('失败' + data2)
})

变量p1是一个Promise对象,它负责执行test函数。由于test函数在内部是异步执行的,当test函数执行成功时,我们告诉Promise对象:

如果执行成功 则执行

p1.then(function (data) {
console.log('成功' + data)
});

失败 则执行:

p2.catch(function (data2) {
console.log('失败' + data2)
})

Promise对象可以串联起来,所以上述代码可以简化为:

new Promise(test).then(function (data) {
console.log('成功:' + data);
}).catch(function (data2) {
console.log('失败:' + data2);
});

继续上面的栗子 看看promise是如何实现异步执行的:

new Promise(function (success, fail) {
console.log('start new Promise...');
var timeOut = Math.random() * 2;
console.log('set timeout to: ' + timeOut + ' seconds.');
setTimeout(function () {
if (timeOut < 1) {
console.log('执行成功函数');
success('200 OK');
}
else {
console.log('执行失败函数');
fail('timeout in ' + timeOut + ' seconds.');
}
}, timeOut * 1000);
}).then(function (r) {
console.log('Done: ' + r);
}).catch(function (reason) {
console.log('Failed: ' + reason);
});

可以将上面的代码打入控制台  你可以看到类如下面的结果

可见Promise最大的好处是在异步执行的流程中,把执行代码和处理结果的代码清晰地分离了

此时 我们再看一个axios的请求  axios支持promise API

axios.get('/user?ID=12345')
.then(function (response) {
console.log(response);
})
.catch(function (error) {
console.log(error);
});

怎么样 形式上 是不是就会感觉到 一种很熟悉的,这样也就能理解 axios请求的写法了。

Promise还可以做更多的事情,比如,有若干个异步任务,需要先做任务1,如果成功后再做任务2,任何任务失败则不再继续并执行错误处理函数。

要串行执行这样的异步任务,不用Promise需要写一层一层的嵌套代码。有了Promise,我们只需要简单地写:

job1.then(job2).then(job3).catch(handleError);

下面有个具体的例子说明:

// 0.5秒后返回input*input的计算结果:
function multiply(input) {
return new Promise(function (resolve, reject) {
console.log('计算 ' + input + ' x ' + input + '...');
setTimeout(resolve, 500, input * input);
});
} // 0.5秒后返回input+input的计算结果:
function add(input) {
return new Promise(function (resolve, reject) {
console.log('计算 ' + input + ' + ' + input + '...');
setTimeout(resolve, 500, input + input);
});
} var p = new Promise(function (resolve, reject) {
console.log('开始promise');
resolve(123);
}); p.then(multiply)
.then(add)
.then(multiply)
.then(add)
.then(function (result) {
console.log('最后的结果 ' + result);
});

在控制台打出上面的代码 可以看到;

我们可以再看一下 vuejs 官方提供的与服务器通信的插件 axios 的大概实现原理。这里是基于原生ajax请求改写

function ajax(method,url,params) {
var request = new XMLHttpRequest();
return new Promise(function (success,fail) {
request.onreadystatechange = function () {
if(request.readyState === 4 && request.status === 200) {
success(request.responseText);
}else {
fail(request.status)
}
}
request.open(method,url);
request.send(params)
})
}

紧接着 我们的调用

var p = ajax('GET', '/api',params);
p.then(function (text) { // 如果AJAX成功,获得响应内容
console.log(text)
}).catch(function (status) { // 如果AJAX失败,获得响应代码
lconsole.log(status);
});

另外:

promise.all()方法,可以支持并行异步操作

var p1 = new Promise(function (resolve, reject) {
setTimeout(resolve, 500, '第一个');
});
var p2 = new Promise(function (resolve, reject) {
setTimeout(resolve, 600, '第二个');
});
// 同时执行p1和p2,并在它们都完成后执行then:
Promise.all([p1, p2]).then(function (results) {
console.log(results); // 获得一个Array: ['第一个', '第二个']
});

Promise.race()方法,可以实现两个异步请求,有一个先返回即可

            var p1 = new Promise(function (resolve, reject) {
setTimeout(resolve, 500, '第一个');
});
var p2 = new Promise(function (resolve, reject) {
setTimeout(resolve, 600, '第二个');
});
// 同时执行p1和p2,并在它们都完成后执行then:
Promise.race([p1, p2]).then(function (results) {
console.log(results); // 输出第一个 第二个会被默认丢弃
});

由于p1运行较快,所以promise的then 会获得 ‘第一个’的输出。第二个会被默认丢弃。

综上:

如果我们组合使用Promise,就可以把很多异步任务以并行和串行的方式组合起来执行

Promise的一些相关讲解的更多相关文章

  1. JWT的相关讲解

    原文链接: http://www.cnblogs.com/chenwolong/p/Token.html

  2. 关于git rebase的相关讲解

    http://gitbook.liuhui998.com/4_2.html 一.基本 git rebase用于把一个分支的修改合并到当前分支. 假设你现在基于远程分支"origin" ...

  3. 关于html的相关讲解

    浏览器chrome Chrome它内部有一个解析器,这个解析器就是解析我们的代码,各个浏览器的内核不一样,所以存在浏览器的兼容.这个内核是一个引擎. 谷歌的内核是webkit 引擎是v8. 客户端的请 ...

  4. Promise原理讲解 && 实现一个Promise对象 (遵循Promise/A+规范)

    1.什么是Promise? Promise是JS异步编程中的重要概念,异步抽象处理对象,是目前比较流行Javascript异步编程解决方案之一 2.对于几种常见异步编程方案 回调函数 事件监听 发布/ ...

  5. 【Mocha.js 101】同步、异步与 Promise

    前情提要 在上一篇文章<[Mocha.js 101]Mocha 入门指南>中,我们提到了如何用 Mocha.js 进行前端自动化测试,并做了几个简单的例子来体验 Mocha.js 给我们带 ...

  6. 浅谈Promise

    学习过JavaScript的人都知道,JavaScript是单线程作业,这样会有一个很大的缺陷,所有的Ajax,浏览器事件等,都是通过异步去完成.所谓的同步和异步最大的区别无非就是在于同步会阻塞后续代 ...

  7. 分步理解 Promise 的实现

    一个 Promise 的运用: var firstPromise = new Promise(function(resolve,reject){ setTimeout(function(){ var ...

  8. ES6入门八:Promise异步编程与模拟实现源码

    Promise的基本使用入门: ——实例化promise对象与注册回调 ——宏任务与微任务的执行顺序 ——then方法的链式调用与抛出错误(throw new Error) ——链式调用的返回值与传值 ...

  9. promise 进阶 —— async / await 结合 bluebird

    一.背景 1.Node.js 异步控制 在之前写的 callback vs async.js vs promise vs async / await 里,我介绍了 ES6 的 promise 和 ES ...

随机推荐

  1. Python list降序排序

    test = [6,1,2,3,4,5] a = sorted(test,reverse=True) print a 结果如下: [6, 5, 4, 3, 2, 1] 你可以参考下sorted,里面是 ...

  2. UVA-11478 Halum (差分约束系统)

    题目大意:一张n个节点的有向带边权图,每次操作能任选一个节点v个一个整数d,使以v为终点的边权值都减少d,以v为起点的边权值都增加d,求若干次操作后的最小边权值的非负最大值. 题目分析:用sum[i] ...

  3. js排序算法02——插入排序

    插入排序的思路是我们默认数组的第一个元素是有序的,从第二个元素开始依次和前面的元素比较,如果前面的元素大,就将前面的元素往后移一位,如果前面的元素小,就把该元素放在前面元素的后面.其实就和我们玩扑克牌 ...

  4. line-height:150% 和 line-height:1.5的区别

    line-height:150% 是继承父元素的距离 line-height:1.5  是计算各子元素的距离 1.当line - height 为百分比时: body{ font-size:14px; ...

  5. jquery中ajax跨域的写法

    由于JS同源策略的影响,因此js只能访问同域名下的文档.因此要实现跨域,一般有以下几个方法: 一.处理跨域的方式: 1.代理 2.XHR2 HTML5中提供的XMLHTTPREQUEST Level2 ...

  6. JSP里的<c:if>不起作用[待解答]

    JSP页面的部分代码如下: 下面的title作为请求参数,shoppingCart作为session范围域的属性. 问题1: 如果去掉<c:if>的判断条件,第一行打印:可以正常显示出来, ...

  7. 数据仓库(Data Warehouse)建设

    数据仓库初体验 数据库仓库架构以前弄的很简单:将各种源的数据统一汇聚到DW中,DW没有设计,只是将所有数据汇聚起来: ETL也很简单,只是将数据同步到DW中,只是遇到BUG时,处理一些错误数据,例如: ...

  8. vue做路由页面内容跳转

    安装----npm npm install vue-router 如果在一个模块化工程中使用它,必须要通过 Vue.use() 明确地安装路由功能: import Vue from 'vue' imp ...

  9. Git详解之七 自定义Git

    以下内容转载自:http://www.open-open.com/lib/view/open1328070404827.html自定义 Git 到目前为止,我阐述了 Git 基本的运作机制和使用方式, ...

  10. Matlab以MEX方式调用C源代码【转载】

    原文地址:http://blog.sina.com.cn/s/blog_468651400100coas.html 这是自己整理的一个对应的文档:<Matlab以MEX方式调用C源代码> ...