jQuery(五): Deferred

有啥用

通常来说,js请求数据,无论是异步还是同步,都不会立即获取到结果,通常而言,我们一般是是使用回调函数再执行,而 deferred就是解决jQuery的回调函数方案,总的来说,deferred对象就是为了将某个回调函数延迟到某个时机再执行.

  1. ajax链式写法:

      //一般写法:
    $.ajax({
    url: '',
    success: function(){},
    error: function(){},
    }) //deferred
    $.ajax(url)
    .done(function(){}) //相当于success
    .fail(function(){})
  2. 指定同一操作的多个函数,允许添加多个函数

    写法也很简单,直接添加在后面就可以了。

     $.ajax(url)
    .done(function(){})
    .fail(function(){})
    .done(function(){})
  3. 为多个函数添加指定回调,可以为多个不同的函数添加同一个回调事件

      $.when($.ajax(url),$.ajax(url2))
    .done()
    .fail()

    为两个函数执行操作,如果都成功了就执行done中的回调,如果有一个失败或全部都失败,就执行fail中的回调

  4. 普通操作的回调

    deferred允许任何操作都可以使用deferred对象的方法,指定回调函数

     var wait = function(de){
    var test = function(){
    console.log('开始');
    de.resolve();
    }
    setTimeout(test, 3000);
    return de;
    } $.when(wait($.Deferred()))
    .done(function(){
    console.log('已完成')
    })
    .fail(function(){
    console.log('失败')
    })

注意: $.when()的参数只能是deferred对象。

咋处理

  1. 关于resolve && rejected

    在上面的时候,会注意到一个resolve,并且会觉得这种链式写法很眼熟,且对promise有一个简单了解的话,大概就知道了。

       promise: 同样也是用于处理异步函数,将异步操作队列化处理
    
       简单的promise
    new promise (function(resolve,rejected){
    resolve('成功')
    })
    .then(function(){}) promise.then 接受两个参数:
    一、 resolve 代表成功时调用的函数
    二、 rejected 代表失败时调用的回调
    promise的三个状态值: pending(初始状态值), fulfilled(操作成功),rejected(操作失败)

    $.deferred 同样也是有三个不同的状态:未完成,已完成,已失败,当状态处于已完成(resolve)下回自动调用done()中的回调函数,而resolve()就是人为将状态值修改为已完成,同理可证rejected();

    总的来说,核心就是:根据不同的状态值调用回调。

  2. API

    * $.Deferred()

    * $.when()

    * deferred.progress()

    * deferred.promise()

    * deferred.done()

    * deferred.fail()

学习下

来看下jQ的源码是怎么处理的:

    Deferred: function(func) {
var tuples = [
// action, add listener, callbacks,
// ... .then handlers, argument index, [final state]
["notify", "progress", jQuery.Callbacks("memory"),
jQuery.Callbacks("memory"), 2
],
["resolve", "done", jQuery.Callbacks("once memory"),
jQuery.Callbacks("once memory"), 0, "resolved"
],
["reject", "fail", jQuery.Callbacks("once memory"),
jQuery.Callbacks("once memory"), 1, "rejected"
]
],
state = 'pending', // 延迟对象
deferred = {},
promise = {
state: function() {
return state
},
then: function(){},
promise: function(obj) {
return obj != null ? jQuery.extend(obj, promise): promise;
}
}
...
}

从代码来看,定一个了数组tuples,以及初始状态值。tuples存储了三个状态下的所需参数,来看下存储了写什么内容:

[状态, 对应的处理函数, 利用callbacks创建的回调队列, then方法的回调队列, index, 最终的状态值],

我们可以看到最终的状态值只有reject 和resolve才有。

ok,已经知道deferred的本质是根据不同的状态调用不同的方法,并且使用callbacks添加函数,那么把tuples遍历一下,生成队列;

源码:

    tuples.forEach(function(tuple){
var list = tuple[2], // 获取到jQuery.callbacks返回,创建一个队列
stateString = tuple[5], //获取到最终状态描述 //promise[ progress | done | fail ] = list.add
promise[tuple[1]] = list.add; // 如果最终状态值存在,即处于 reject|| resolve 状态下;
if (stateString) {
list.add(
function() {
state = stateString;
}
....
)
}
// 延迟对象状态 deferred.resolve()
//deferred[ 'resolve' | reject | notify] = function(){} deferred[tuple[0]] = function() {
deferred[tuple[0]+"Width"](this === deferred ? promise : this, arguments);
return this;
} //jQuery.callbacks.fireWith
//执行队列,调用处理函数,绑定执行时的上下文
deferred[tuple[0] + "With"] = list.fireWith;
})
promise.promise(deferred);
return deferred;

已经遍历生成了3个队列,并将三个状态方法挂载在了延迟对象上。

从代码中可以看出,在调用deferred[ reject | resolve]时,其实是调用了deferred[ rejectWith | resolveWith]方法,本质上是对callbacks.fireWith的调用,以用来执行添加的回调函数,同时设置函数的上下文。

并且可以看的到,deferred[proress | done | fail] 其实是copy了callbacks.add方法,将回调函数添加在了执行队列中。

另外关于对jQuery.deferred对象的详解,使用,https://www.cnblogs.com/chris-oil/p/8922770.html 这篇博文转载了阮大神的,可以瞅瞅

jQuery(五): Deferred的更多相关文章

  1. jQuery之Deferred源码剖析

    一.前言 大约在夏季,我们谈过ES6的Promise(详见here),其实在ES6前jQuery早就有了Promise,也就是我们所知道的Deferred对象,宗旨当然也和ES6的Promise一样, ...

  2. jQuery的deferred对象详解(一)

    最近一段时间,都在研究jquery里面的$.Deffered对象,几天都搞不明白,其中源码的运行机制,网上查找了相关的资料,<jQuery的deferred对象详解>阮一峰老师的文章,里面 ...

  3. [转] jQuery的deferred对象详解

    jQuery的开发速度很快,几乎每半年一个大版本,每两个月一个小版本. 每个版本都会引入一些新功能.今天我想介绍的,就是从jQuery 1.5.0版本开始引入的一个新功能----deferred对象. ...

  4. jQuery的deferred对象详解(转)

    jQuery的开发速度很快,几乎每半年一个大版本,每两个月一个小版本. 每个版本都会引入一些新功能.今天我想介绍的,就是从jQuery 1.5.0版本开始引入的一个新功能----deferred对象. ...

  5. jQuery的deferred对象详解 jquery回调函数

    http://www.ruanyifeng.com/blog/2011/08/a_detailed_explanation_of_jquery_deferred_object.html jQuery的 ...

  6. jQuery的deferred对象详解(转载)

    jQuery的开发速度很快,几乎每半年一个大版本,每两个月一个小版本.(由于无法转载,复制原文 .原文链接——原作者:阮一峰) 每个版本都会引入一些新功能.今天我想介绍的,就是从jQuery 1.5. ...

  7. 【转】jQuery的deferred对象详解

    jQuery的开发速度很快,几乎每半年一个大版本,每两个月一个小版本. 每个版本都会引入一些新功能.今天我想介绍的,就是从jQuery 1.5.0版本开始引入的一个新功能----deferred对象. ...

  8. jQuery的deferred对象详解(转)

    jQuery的开发速度很快,几乎每半年一个大版本,每两个月一个小版本. 每个版本都会引入一些新功能.今天我想介绍的,就是从jQuery 1.5.0版本开始引入的一个新功能----deferred对象. ...

  9. (转)jQuery的deferred对象详解

    作者: 阮一峰 日期: 2011年8月16日 jQuery的开发速度很快,几乎每半年一个大版本,每两个月一个小版本. 每个版本都会引入一些新功能.今天我想介绍的,就是从jQuery 1.5.0版本开始 ...

随机推荐

  1. git常用命令(测试必备)

    什么是git 百度百科:Git(读音为/gɪt/)是一个开源的分布式版本控制系统,可以有效.高速地处理从很小到非常大的项目版本管理. git相对svn,有很多优势,这里就不再赘述,大家网上搜索吧. g ...

  2. Resource Site

    GameDevMarket.net Collection of Music, Sound effects (sfx), 2D/3D/GUI Art. http://gaurav.munjal.us/U ...

  3. #pragma once用法总结

    1.#pragmaonce这个宏有什么作用? 为了避免同一个头文件被包含(include)多次,C/C++中有两种宏实现方式:一种是#ifndef方式:另一种是#pragma once方式. 在能够支 ...

  4. Pandas | 09 迭代

    Pandas对象之间的基本迭代的行为取决于类型.当迭代一个系列时,它被视为数组式,基本迭代产生这些值.其他数据结构,如:DataFrame和Panel,遵循类似惯例,迭代对象的键. 简而言之,基本迭代 ...

  5. Nacos

    欢迎来到 Nacos 的世界! Nacos 致力于帮助您发现.配置和管理微服务.Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现.服务配置.服务元数据及流量管理. Nacos 帮助您 ...

  6. HTML 超链接返回上一级

    参考:http://blog.csdn.net/huanongjingchao/article/details/39587663 超链实现返回刚刚访问的网页: <a href="#&q ...

  7. NOIP2013-2014提高组题目浅析

    1.前言 迎接NOIP的到来...在这段闲暇时间,决定刷刷水题.这里只是作非常简单的一些总结. 2.NOIP2014 <1> 生活大爆炸之石头剪刀布(模拟) 这是一道考你会不会编程的题目. ...

  8. display:none和visibility:hidden

    w3school学习网:https://www.w3school.com.cn/tiy/t.asp?f=hdom_style_display_none display: none----将元素的显示设 ...

  9. quick 中 "我的项目" 中的列表从那里来的?

    quick 中 "我的项目" 中的列表从那里来的? 1. WelcomeScene.lua 场景 self:createOpenRecents(cc.player.settings ...

  10. Linux学习之编译运行.c(C语言)文件

    在Linux命令行界面下,创建文件hello.c,进入vim编辑器,编辑一个简单的C语言文件 分解C语言文件执行过程,要经过预编译.编译.汇编.连接四个步骤后才能执行, 预编译:gcc -E hell ...