Promise使用

1. 概述

在JavaScript的世界中,所有代码都是单线程执行的,在浏览器中,网络请求需要异步请求,出现了ajax技术,Promise辅助我们做异步操作,所谓promis就是“承诺未来会执行”,其实我们更多是用做“先写好任务,再某个时机获取执行结果”

2. 入门

2.1 创建Promise

<script>    
    let promise = new Promise(function() {
        console.log('进入了Promise');
    });
    
    console.log('end'); 
</script>
执行结果

我们发现只要创建了promise,会==立即执行它内部的回调函数==

进入了Promise
end

2.2 解决Promise创建即执行问题

promise回调函数要做的事情,很多时候并不是要立刻执行,比如我们封装了一个网络请求,我们要点击的时候才发出去,这就是个问题了,如果解决?

答:使用函数返回值来解决这个问题!!!

<script>    
    let fn = function() {
        return new Promise(function() {
            console.log('进入了Promise');
        });
    }
    
    console.log('end'); 
    
    fn();   
</script>
执行结果

执行了fn函数,相当于把Promise创建return出来,而创建的同时,相当于执行了Promise内部的回调函数

end
进入了Promise

2.3 模拟用Promise封装网络请求

  1. Promise构造函数可以接受一个回调函数,回调函数可以接受2个参数

第一个代表:成功的回调函数resolve;第二个代表:失败的回调函数reject

  1. Promise实例对象获执行结果,可以通过2个方法

then方法:接收resolve函数执行结果;catch:接收reject函数执行结果

<script type="text/javascript">
    
    // 1. 模拟把要做的事情先封装起来
    var promise = new Promise(function(resolve, reject){
        console.log('执行promise');
                
        // 模拟请求回调
        let flag = parseInt(Math.random()*2) == 1;
        if (flag) {
            setTimeout(resolve, 2000, '请求成功');
        }else {
            setTimeout(reject, 2000, '请求失败');
        }
    });
    
    // 2. 模拟执行网络请求,拿到请求结果
    promise.then(function(res){
        // 成功被这个函数接收
        console.log(res);
    }).catch(function(res){
        // 失败被这个函数接收
        console.log(res);
    }); 
    
    console.log('结束');  
</script>
执行结果

未将Promise用函数包起来,HTML一加载到这里就会立即执行网络请求,而不是适当时机,还要优化

执行promise
结束
请求成功

2.4 使用Promise完善网络请求封装

准备一个Java测试接口
@WebServlet("/test")
public class TestServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

        ResultInfo resultInfo = new ResultInfo();
        // 随机成功或失败
        if (ThreadLocalRandom.current().nextBoolean()) {
            resultInfo.setFlag(true);
            resultInfo.setErrorMsg("请求成功");
        }else {
            resultInfo.setFlag(false);
            resultInfo.setErrorMsg("请求失败");
        }

        response.setContentType("application/json;charset=utf-8");
        ObjectMapper objectMapper = new ObjectMapper();
        objectMapper.writeValue(response.getWriter(), resultInfo);
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
}
前端代码
<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8" />
        <title></title>
    </head>
    <body>
        <button type="button" id="btn">发起请求</button>
        <div id="info">请求结果:</div>
    </body>
</html>

<script>
    // 1. ajax函数将返回Promise对象:
    let ajax = function (method, url, data) {
        var request = new XMLHttpRequest();
        return new Promise(function (resolve, reject) {
            request.onreadystatechange = function () {
                if (request.readyState === 4) {
                    if (request.status === 200) {
                        let res = JSON.parse(request.response);
                        if (res['flag']) {
                            // 请求成功
                            resolve(request.responseText);
                        }else {
                            // 接口反馈请起有问题
                            reject(request.responseText);
                        }
                    } else {
                        // 网络问题
                        reject(request.status);
                    }
                }
            };
            request.open(method, url);
            request.send(data);
        });
    }
    
    // 2. 点击按钮发起请求
    document.getElementById("btn").onclick = function() {
        ajax('POST', 'http://192.168.142.129/travel/test', {'key': 'is a request param'})
        .then(function(res) {
            console.log('请求成功数据:' + res);
        }).catch(function(res) {
            console.log('请求失败数据:' + res);
        });
    }
    
</script>

3. 其他用法

3.1 合并任务All

有时候可能某个任务,需要前几个任务完成后,才能执行这个任务

比如:多线程请求各自下载一个大图片某一部分,最终确保都执行完毕后,再执行合并图片的任务,这个任务执行时机取决于最后一个完成任务的线程

<script>    
    let promise1 = new Promise(function(resolve, reject){
        setTimeout(resolve('promise1执行完毕'), 1000);
    });
    
    let promise2 = new Promise(function(resolve, reject){
        // 定时器另外写法,传参写后面
        setTimeout(resolve, 5000, 'promise2执行完毕');
    });
    
    // 合并执行
    Promise.all([promise1, promise2]).then(function(results){
        console.log(results);
    }); 
</script>
执行结果

5秒钟后执行回调,回调函数结果是多个promis传入的结果合并成数组

['promise1执行完毕', 'promise2执行完毕']

3.2 竞赛任务Race

有时候可能多个类似任务执行,只要有一个先执行完就可以

比如:只录取第一名,大家都做一样的事情,但是谁先赢了,其他人也不用比了

<script>    
    let promise1 = function(){
        return new Promise(function(resolve, reject){
            // 定时器另外写法,传参写后面
            setTimeout(resolve, 5000, 'promise1执行完毕');
        })
    };
    
    let promise2 = function(){
        return new Promise(function(resolve, reject){
            // 定时器另外写法,传参写后面
            setTimeout(resolve, 3000, 'promise2执行完毕');
        })
    };
    
    // 合并执行
    Promise.race([promise1(), promise2()]).then(function(result){
        console.log(result);
    }); 
</script>
执行结果

直接执行了promise2,其实promise1也是继续执行,只是结果被丢弃了

promise2执行完毕

4. 参考资料

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容