异步

JS中的单线程指的是在同一个时间内只能做一件事情。

好处:避免DOM渲染的冲突。浏览器根据HTML文件初始化需要渲染DOM,JS也可以修改DOM,为了避免两种方式对同一个DOM元素的处理发生冲突,JS执行时,渲染要暂停处理,除此之外,两段JS代码也不能同时执行。

引入的问题:造成页面的卡顿。当页面渲染过程中,执行JS加载某一段数据,需要等到数据加载完毕后浏览器才能继续渲染操作,导致用户体验很差。

如何解决单线程处理的缺陷?——异步

1、setTimeOut(callback , [time])

直接跳过该段代码,继续执行下面的主线程代码,time毫秒之后把callback函数放入到异步队列中(或者省略掉time,直接把callback函数放入到异步队列)。当主线程的代码执行完毕后,再在异步队列中取函数执行。

2、$.ajax( {url:'xxxxx' ,success:callback })

当遇到ajax代码需要从其他页面加载数据时,也是跳过这段代码,继续在主线程中执行JS代码,当url地址的数据加载完毕后,把callback函数放入异步队列,等待主线程执行完毕后的回调。

好处:释放了加载数据这段空期,代码能够继续执行。

引入的问题:异步的操作没有按照书写的方式执行,可读性差。而且,callback中不容易模块化,当新增需求,只能在callback函数中补充功能的实现,这就违反了开放封闭原则,也不利于测试和维护。

异步的本质——实现方式——event-loop

同步代码在主线程中按次立即执行,遇到异步函数时则略过,当某些事件被触发(比如定时器或者数据加载完毕),则把异步函数放入到异步队列中。最后,当同步代码执行完毕,则启用事件轮询机制,如果在队列中有异步函数,则取出来放入到主线程中运行,如果是空队列,则继续轮询监听异步队列中是否有新的异步函数被插入。

如何解决异步引入的问题?——deferred

这里从jQuery1.5版本前后ajax的特性说起。

jQuery1.5版本前的ajax语法:

var ajax = $.ajax({

      url:'xxxxxxx',

      success:function(){

          xxxxxxxxxxxxxxx

      },

    error:function(){

          xxxxxxxxxxxxxxx

      }   

})

//返回的是一个XHR对象

使用这种方式,当需要回调函数success的代码时,不得不在success代码内部添加,严重影响了代码的安全性。所以,在之后的jQuery1.5版本以后,对这种行为进行了规避:

var ajax = $.ajax('url')

ajax.done(function(){xxxxxx}).fail(function(){xxxxxx}).done(function(){......})......

//返回一个deferred对象

或者:

ajax.then(function(){xxx},function(){xxx}).then(function(){xxx},function(){xxx})...

//这边引入的ajax.then格式就和之后提出的promise标准很相近了

使用ajax链式操作扩展success或者error代码。

jQuery1.5的变化,从本质上来说它无法改变JS异步和单线程的本质,实际上它是一种语法糖的形式,但是解耦了代码,从写法上杜绝了callback这种形式,很好地体现了开放封闭原则。

如何使用jQuery Deferred?

如下是一段简单的异步操作代码,不利于扩展操作:

var wait = function () {

          var task = function () {

                  console.log('执行完毕')

          }

          setTimeOut(task,2000)

}

wait()

现在新增需求,需要在执行完毕之后进行某些特别复杂的操作,而且可能分为好几个步骤,这里就需要引入deferred对象来进行拓展:

function waitHandle () {

      //创建一个deferred对象

        var dtd = $.Deferred()

      //要求传入一个deferred对象

        var wait = function (dtd) {

                  var task = function () {

                          console.log('执行完毕')

                        //表示异步任务已经完成

                          dtd.resolve()

                          //表示异步任务失败或出错

                          dtd.reject()

                  }

                  setTimeOut(task,2000)

                //要求返回deferred对象

                return dtd

        }

        //注意:这里一定要有返回值

        return wait(dtd)

}

//返回deferred对象

var w = waitHandle()

//调用deferred对象的then方法。then方法传入两个函数

w.then(function(){

            console.log(‘ok 1’)

},function(){

            console.log(‘error 1’)

}).then(function(){

            console.log(‘ok 2’)

},function(){

            console.log(‘error 2’)

})

promise对象最初的引入——Deferred.promise()——分离API

如上述代码所示,dtd的API分为两大类,它们的用意不同。

第一类是主动触发的:dtd.resolve(),dtd.reject()

第二类是被动监听的:dtd.then(),dtd.done(),dtd.fail()

如果在then前面主动触发resolve或者reject,后面负责监听的API会根据距离最近的监听结果来执行自己相应的代码,发生意想不到的结果。

为了把这两类API分开,引入deferred.promise()对象:

var wait = function (dtd) {

                  var task = function () {

                          console.log('执行完毕')

                          dtd.resolve()

                          dtd.reject()

                  }

                  setTimeOut(task,2000)

                //注意这里返回的是promise,而不是deferred对象

                return dtd.promise(()

        }

                ..............

//经过上面的改动,w接收的是一个promise对象

var w = waitHandle()

$.when(w)

  .then(function(){

          console.log(‘ok 1’)

  })

  .then(function(){

          console.log(‘ok 2’)

  })

//此处若执行w.reject会报错,因为promise对象中没有这个方法

ES6中的Promise


浅谈Promise的实现:

Promise 对象用于表示一个异步操作的最终状态(完成或失败),以及其返回的值

一些旧版浏览器不支持Promise,可以在<script>中引入bluebird

异常捕获规定:then()只能接收一个参数,最后统一由catch捕获异常。

catch不仅能捕获程序逻辑之外语法的错误,还能捕获业务逻辑之内错误,reject需要传入参数,不应该为空参数

result.then(function (img) {

        alert(img.width)

}).then(function(img) {

        alert(img.height)

}).catch(function(ex) {

        alert(ex)

})

多个串联

应用场景:按照顺序执行

var src1 = './imgDemo1.png'

var result1 = loadImg(src1)

var src2 = './imgDemo2.png'

var result2 = loadImg(src2)

result1.then(function (img) {

        alert(img.width)

        return result2

}).then(function(img) {

        alert(img.height)

}).catch(function(ex) {

        alert(ex)

})

Promise.all & Promise.race

Promise.all接收一个promise对象的数组

待全部完成之后,统一执行success:

Promise.all([ result1 ], [ result2 ]).then(datas=> {

    //接收到的datas是一个数组,依次包含多个promise返回的内容

    console.log(datas[ 0 ])

    console.log(datas[ 1 ])

})

Promise.all可以将多个Promise实例包装成一个新的Promise实例。同时,成功和失败的返回值是不同的,成功的时候返回的是一个结果数组,而失败的时候则返回最先被reject失败状态的值。

Promise.race接收一个包含多个promise对象的数组

只要有一个完成,就执行success

Promise.all([ result1 ], [ result2 ]).then(data=> {

//接收到的data是最先执行完的promise的返回值

    console.log(data)

})

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

推荐阅读更多精彩内容

  • -al -ial/-ual/-ical/-el/-le 主要做形容词,还可以作名词。 ❈来源于中古英语中的-al和...
    单词家族阅读 4,853评论 0 0
  • 我最想实现的一个愿望, 是找到一条永生不死的路, 包括身体和心灵, 尽管这一点也不着边际, 但是我就是在这样不切实...
    D木头仁阅读 413评论 0 3
  • ​ 导语:为什么做销售?别以为能说就能做好销售,更别以为能吹就能出单!电话销售更是不能这样想当然的认为。 老鱼早前...
    0c80af0c57a1阅读 734评论 0 1
  • 一些想法:这次真的终了! 我写的,所分享的东西都是我看到,听到,以及胡思乱想想到的东西;我只是把触动我的内容转述出...
    haominzeng阅读 456评论 0 0
  • ,,,,,,,
    卫凯旋2410阅读 147评论 0 0