ES6学习笔记五(promise异步)

知识点1:rosolve是执行下一步then()

// Promise
{
    let ajax=function(){
        console.log('执行2');
        return new Promise(function(resolve,reject){
            setTimeout(function(){
                resolve()
            },1000);
        })
    }

    ajax().then(function(){
        console.log('promise','timeout2');
    })
}

知识点2:允许多个下一步(then()里面再嵌Promise实例)

{
    // 允许多个下一步
    let ajax=function(){
        console.log('执行3');
        return new Promise(function(resolve,reject){
            setTimeout(function(){
                resolve() //允许下一步then
            },1000);
        })
    }

    ajax()
       .then(function(){
        return new Promise(function(resolve,reject){
            setTimeout(function(){
                resolve();
            },2000);
        });
    })
       .then(function(){
           console.log('timeout3');
       })
}

知识点3:抛错时处理

{
    // 抛错时处理
    let ajax=function(num){
        console.log('执行4');
        return new Promise(function(resolve,reject){
            if(num>5){
                resolve()
            }else {
                throw new Error('出错了')
            }
        })
    }

    ajax(6).then(function(){
        console.log('log',6);
    }).catch(function(err){
        console.log('catch',err);
    });

    ajax(3).then(function(){
        console.log('log',3);
    }).catch(function(err){
        console.log('catch',err);
    });
}

应用场景:图片加载( Promise.all([]).then() 和  Promise.race([]).then() )

场景1:多张图片加载,实现所有图片加载完,再一起出现在页面上,提高用户体验

{

    // 所有图片加载完再添加到页面
    function loadImg(src){
        return new Promise((resolve,reject)=>{
            let img=document.createElement('img');
            img.src=src;
            img.onload=function(){
                resolve(img);
            }
            img.onerror=function(err){
                reject(err);
            }
        })
    }

    function showImgs(imgs){
        imgs.forEach(function(img){
            document.body.appendChild(img);
        })
    }

    Promise.all([  //all是把多个Promise实例,只有所有合成一个Promise实例,只有所有Promise实例状态都发生变化时,才会生成新的Promise实例
        loadImg('https://cdn.baigebao.com/upload/pc/20181016/5bc53a30b2291.png'),
        loadImg('https://cdn.baigebao.com/upload/pc/20181008/5bbad45787a22.png'),
        loadImg('https://cdn.baigebao.com/upload/pc/20181016/5bc53a30b2291.png')
    ]).then(showImgs);

}

场景2:有一个图片加载完成就添加到桌面

{
    // 有一个图片加载完成就添加到桌面
    function loadImg(src){
        return new Promise((resolve,reject)=>{
            let img=document.createElement('img');
            img.src=src;
            img.onload=function(){
                resolve(img);
            }
            img.onerror=function(err){
                reject(err);
            }
        })
    }

    function showImgs(img){
        let p=document.createElement('p');
        p.appendChild(img);
         document.body.appendChild(p);
    }

    Promise.race([  //只要有一个状态改变就先显示,只显示一张
        loadImg('https://cdn.baigebao.com/upload/pc/20181016/5bc53a30b2291.png'),
        loadImg('https://cdn.baigebao.com/upload/pc/20181008/5bbad45787a22.png'),
        loadImg('https://cdn.baigebao.com/upload/pc/20181016/5bc53a30b2291.png')
    ]).then(showImgs);

}

 

注意:

promise中的resolved 总是晚于本轮循环中的同步任务
例子:
new Promise((resolve, reject) => {
  resolve(1);
  console.log(2);
}).then(r => {
  console.log(r);
});
// 2
// 1

调用resolve(1)以后,后面的console.log(2)还是会执行,并且会首先打印出来。这是因为立即 resolved 的 Promise 是在本轮事件循环的末尾执行,总是晚于本轮循环的同步任务。

一般来说,调用resolve或reject以后,Promise 的使命就完成了,后继操作应该放到then方法里面,而不应该直接写在resolve或reject的后面。所以,最好在它们前面加上return语句,这样就不会有意外。

new Promise((resolve, reject) => {
  return resolve(1);
  // 后面的语句不会执行
  console.log(2);
})

 

 
posted @ 2018-11-09 15:41  心向阳  阅读(291)  评论(0编辑  收藏  举报