一、主要用于异步计算
二、能够将异步操做队列化,按照指望的顺序执行,返回符合预期的结果
三、能够在对象之间传递和操做promise,帮助咱们处理队列html
为了不界面冻结(任务)node
同步:假设你去了一家饭店,找个位置,叫来服务员,这个时候服务员对你说,对不起我是“同步”服务员,我要服务完这张桌子才能招呼你。那桌客人明明已经吃上了,你只是想要个菜单,这么小的动做,服务员却要你等到别人的一个大动做完成以后,才能再来招呼你,这个即是同步的问题:也就是“顺序交付的工做1234,必须按照1234的顺序完成”。python
异步:则是将耗时很长的A交付的工做交给系统以后,就去继续作B交付的工做,。等到系统完成了前面的工做以后,再经过回调或者事件,继续作A剩下的工做。
AB工做的完成顺序,和交付他们的时间顺序无关,因此叫“异步”。jquery
document.getElementById('#start').addEventListener('click', start, false); function start() { // 响应事件,进行相应的操做 } // jquery on 监听 $('#start').on('click', start)
2.回调ajax
// 比较常见的有ajax $.ajax('http://www.wyunfei.com/', { success (res) { // 这里能够监听res返回的数据作回调逻辑的处理 } }) // 或者在页面加载完毕后回调 $(function() { // 页面结构加载完成,作回调逻辑处理 })
你们都知道在nodeJS出来以前PHP、Java、python等后台语言已经很成熟了,nodejs要想可以有本身的一片天,那就得拿出点本身的绝活:
无阻塞高并发,是nodeJS的招牌,要达到无阻塞高并发异步是其基本保障
举例:查询数据从数据库,PHP第一个任务查询数据,后面有了新任务,那么后面任务会被挂起排队;而nodeJS是第一个任务挂起交给数据库去跑,而后去接待第二个任务交给对应的系统组件去处理挂起,接着去接待第三个任务...那这样子的处理必然要依赖于异步操做数据库
通常状况咱们一次性调用API就能够完成请求。
有些状况须要屡次调用服务器API,就会造成一个链式调用,好比为了完成一个功能,咱们须要调用API一、API二、API3,依次按照顺序进行调用,这个时候就会出现回调地狱的问题数组
new Promise( function (resolve, reject) { // 一段耗时的异步操做 resolve('成功') // 数据处理完成 // reject('失败') // 数据处理出错 } ).then( (res) => {console.log(res)}, // 成功 (err) => {console.log(err)} // 失败 )
new Promise(resolve => { setTimeout(() => { resolve('hello') }, 2000) }).then(res => { console.log(res) })
new Promise(resolve => { setTimeout(() => { resolve('hello') }, 2000) }).then(val => { console.log(val) // 参数val = 'hello' return new Promise(resolve => { setTimeout(() => { resolve('world') }, 2000) }) }).then(val => { console.log(val) // 参数val = 'world' })
let pro = new Promise(resolve => { setTimeout(() => { resolve('hello world') }, 2000) }) setTimeout(() => { pro.then(value => { console.log(value) // hello world }) }, 2000)
结论:promise做为队列最为重要的特性,咱们在任何一个地方生成了一个promise队列以后,咱们能够把他做为一个变量传递到其余地方。promise
一、接收两个函数做为参数,分别表明fulfilled(成功)和rejected(失败)
二、.then()返回一个新的Promise实例,因此它能够链式调用
三、当前面的Promise状态改变时,.then()根据其最终状态,选择特定的状态响应函数执行
四、状态响应函数能够返回新的promise,或其余值,不返回值也能够咱们能够认为它返回了一个null;
五、若是返回新的promise,那么下一级.then()会在新的promise状态改变以后执行
六、若是返回其余任何值,则会当即执行下一级.then()服务器
一、由于.then()返回的仍是Promise实例
二、会等里面的then()执行完,再执行外面的闭包
Promise会自动捕获内部异常,并交给rejected响应函数处理。
第一种错误处理
2. 第二种错误处理
第一种状况:
结论:catch也会返回一个promise实例,而且是resolved状态
结论:抛出错误变为rejected状态,因此绕过两个then直接跑到最下面的catch
Promise.all([p1, p2, p3])用于将多个promise实例,包装成一个新的Promise实例,返回的实例就是普通的promise
它接收一个数组做为参数
数组里能够是Promise对象,也能够是别的值,只有Promise会等待状态改变
当全部的子Promise都完成,该Promise完成,返回值是所有值得数组
有任何一个失败,该Promise失败,返回值是第一个失败的子Promise结果
//切菜 function cutUp(){ console.log('开始切菜。'); var p = new Promise(function(resolve, reject){ //作一些异步操做 setTimeout(function(){ console.log('切菜完毕!'); resolve('切好的菜'); }, 1000); }); return p; } //烧水 function boil(){ console.log('开始烧水。'); var p = new Promise(function(resolve, reject){ //作一些异步操做 setTimeout(function(){ console.log('烧水完毕!'); resolve('烧好的水'); }, 1000); }); return p; } Promise.all([cutUp(), boil()]) .then((result) => { console.log('准备工做完毕'); console.log(result); })
let p1 = new Promise(resolve => { setTimeout(() => { resolve('I\`m p1 ') }, 1000) }); let p2 = new Promise(resolve => { setTimeout(() => { resolve('I\`m p2 ') }, 2000) }); Promise.race([p1, p2]) .then(value => { console.log(value) })
常见用法:
异步操做和定时器放在一块儿,,若是定时器先触发,就认为超时,告知用户;
例如咱们要从远程的服务家在资源若是5000ms尚未加载过来咱们就告知用户加载失败
现实中的用法
回调包装成Promise,他有两个显而易见的好处:
一、可读性好
二、返回 的结果能够加入任何Promise队列
实战示例,回调地狱和promise对比:
/*** 第一步:找到北京的id 第二步:根据北京的id -> 找到北京公司的id 第三步:根据北京公司的id -> 找到北京公司的详情 目的:模拟链式调用、回调地狱 ***/ // 回调地狱 // 请求第一个API: 地址在北京的公司的id $.ajax({ url: 'https://www.easy-mock.com/mock/5a52256ad408383e0e3868d7/lagou/city', success (resCity) { let findCityId = resCity.filter(item => { if (item.id == 'c1') { return item } })[0].id $.ajax({ // 请求第二个API: 根据上一个返回的在北京公司的id “findCityId”,找到北京公司的第一家公司的id url: 'https://www.easy-mock.com/mock/5a52256ad408383e0e3868d7/lagou/position-list', success (resPosition) { let findPostionId = resPosition.filter(item => { if(item.cityId == findCityId) { return item } })[0].id // 请求第三个API: 根据上一个API的id(findPostionId)找到具体公司,而后返回公司详情 $.ajax({ url: 'https://www.easy-mock.com/mock/5a52256ad408383e0e3868d7/lagou/company', success (resCom) { let comInfo = resCom.filter(item => { if (findPostionId == item.id) { return item } })[0] console.log(comInfo) } }) } }) } })
// Promise 写法 // 第一步:获取城市列表 const cityList = new Promise((resolve, reject) => { $.ajax({ url: 'https://www.easy-mock.com/mock/5a52256ad408383e0e3868d7/lagou/city', success (res) { resolve(res) } }) }) // 第二步:找到城市是北京的id cityList.then(res => { let findCityId = res.filter(item => { if (item.id == 'c1') { return item } })[0].id findCompanyId().then(res => { // 第三步(2):根据北京的id -> 找到北京公司的id let findPostionId = res.filter(item => { if(item.cityId == findCityId) { return item } })[0].id // 第四步(2):传入公司的id companyInfo(findPostionId) }) }) // 第三步(1):根据北京的id -> 找到北京公司的id function findCompanyId () { let aaa = new Promise((resolve, reject) => { $.ajax({ url: 'https://www.easy-mock.com/mock/5a52256ad408383e0e3868d7/lagou/position-list', success (res) { resolve(res) } }) }) return aaa } // 第四步:根据上一个API的id(findPostionId)找到具体公司,而后返回公司详情 function companyInfo (id) { let companyList = new Promise((resolve, reject) => { $.ajax({ url: 'https://www.easy-mock.com/mock/5a52256ad408383e0e3868d7/lagou/company', success (res) { let comInfo = res.filter(item => { if (id == item.id) { return item } })[0] console.log(comInfo) } }) }) }
转载于:https://www.jianshu.com/p/1b63a13c2701