这周做了个小活动(http://aoqi.100bt.com/zt-2016duanzi/index.html),刚开始时候没看好需求,逻辑都写一块了

最后各种坑要填补,从中也获取了些经验和教训,下面说说这里会用到的$.Deferred;

关于jquery里面的deferred的基本使用方法,阮一峰大婶已经有文章说明了,链接如下:

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

这里就说说里面没提及的吧:

页面中逻辑比较麻烦的就是获奖名单的渲染问题,如下图

这里需要的逻辑是:

1、获取时间,判断是否有活动过了投票时间,没有则不可点击,页面不渲染

2、过来时间后,判断能否获取到获奖名单,获取到就渲染次阶段的页面,获取不到则渲染前一个阶段的页面,

  前一个阶段的页面还是获取不到则往前回溯,如果到第一个阶段还是没有则页面不渲染。

假设四个页面对应的获奖名单是1.html、2.html、3.html、4.html;

假如此时有时间判断活动时间已经在第四阶段,即是前三个阶段都结束,我们的实现逻辑或许是

$.get('3.html').done(function(){
//渲染页面
}).fail(function (argument) {
$.get('2.html').done(function (argument) {
//渲染页面
}).fail(function (argument) {
function (argument) {
$.get('1.html').done(function (argument) {
// 渲染页面
}).fail(function (argument) {
// 不渲染页面
})
}
})
})

这是很不可取的,在每个不同的阶段都要嵌套一次,而且每个请求都要等上一个请求发完才发,太慢了。。。

于是就改成类似下面的

var linkArr = ['1.html','2.html','3.html','4.html']
$nav.each(function(index, el) {
var $self = $(this);
$.get(index+'.html').done(function (argument) {
$self.text('获奖名单出来了')
}).fail(function (argument) {
$self.addClass('graynav').text('敬请期待');
});
});

能“并行”地发出多个请求,看似不错,然后在添加点击渲染事件,点击不同的nav渲染已经出来的相应获奖名单,嗯,这也是逻辑上需要的

$('.aCommon_nav').click(function(event) {
if($(this).hasClass('graynav')){
return false;
}
var i = $('aCommon_nav').index($(this));
$.get(linkarr[i], function(data) {
$inforWrap.html(data)
});
});

这时候,我们只需在“并发”请求结束后调用最后一个可点的nav就完成了。。真赞

可是安装现在这种写法,我们并不能判断是否已经完成了请求,或许需要维护个全局变量num,在每次fail和done再加1,当num等于需要发送的个数之后再调用函数

var linkArr = ['1.html','2.html','3.html','4.html']
var i = 0;
$nav.each(function(index, el) {
var $self = $(this);
if($self.attr('xxx')<=timenum){//由时间判断出的是已经结束的标志
$.get(index+'.html').done(function (argument) {
i++;
$self.text('获奖名单出来了');
if(i==LEN){
//DoClickEvent()
}
}).fail(function (argument) {
i++;
$self.addClass('graynav').text('敬请期待');
if(i==LEN){
//DoClickEvent()
}
});
}
});

此时就已经基本完成了此次逻辑了,但是代码实在太糙,需要改进下,下面就用when来完成吧~

var linkArr = ['1.html','2.html','3.html','4.html'],
deferredArr = [];
$('.aCommon_nav').each(function(index, el) {
var $self = $(this);
if($self.attr('xxx')<=timenum){//由时间判断出的是已经结束的标志
deferredArr.push($.get(index+'.html').done(function (argument) {
$self.text('获奖名单出来了');
}).fail(function (argument) {
$self.addClass('graynav').text('敬请期待');
}))
}
}); $.when.apply(null,deferredArr).always(function(arg){
$('.aCommon_nav:not(.garynav)').last().click();
})

这时候,貌似完成了这个逻辑,but,调试之后发现有时候先执行完done,然后fail然后always然后又done,貌似这顺序有些乱来了。。。

然而我们希望的是先done或者fail最后执行always。

发现原因是绑定顺序导致的,于是发现了两条路走,

第一条,把done和fail逻辑都写在always里面,如下

$.when.apply(null,defferredArr).always(function(arg){
$.each(arguments, function(index, val) {
val.done(function (argument) {
$self.text('获奖名单出来了');
}).fail(function (argument) {
$self.addClass('graynav').text('敬请期待');
})
});
$('.aCommon_nav:not(.garynav)').last().click();
})

这感觉好赞,deferred的回调函数都写一块了,维护起来也开心。

但是当deferredArr里面只有一个元素的时候,发现报错了,好尴尬,只能断点看看,

发现always回调函数里面的arguments竟然是一个数组,第一项是请求返回的数据,第二个是返回状态,第三个是此次请求的deferred对象,

于是我们要加个判断,或加个try.catch包裹着done和fail,代码就不贴了,是在太糊弄了。

第二条路,使用setTimeout的黑魔法

var linkArr = ['1.html','2.html','3.html','4.html'],
deferredArr = [];
$('.aCommon_nav').each(function(index, el) {
var $self = $(this);
if($self.attr('xxx')<=timenum){//由时间判断出的是已经结束的标志
deferredArr.push($.get(index+'.html').done(function (argument) {
$self.text('获奖名单出来了');
}).fail(function (argument) {
$self.addClass('graynav').text('敬请期待');
}))
}
}); $.when.apply(null,deferredArr).always(function(arg){
setTimeout(function (argument) {
$('.aCommon_nav:not(.garynav)').last().click();
},0)
})

于是这个业务也完成了,废话也讲完了

												

jquery Deferred使用经验的更多相关文章

  1. JS魔法堂:jQuery.Deferred(jQuery1.5-2.1)源码剖析

    一.前言 jQuery.Deferred作为1.5的新特性出现在jQuery上,而jQuery.ajax函数也做了相应的调整.因此我们能如下的使用xhr请求调用,并实现事件处理函数晚绑定. var p ...

  2. 深入分析,理解jQuery.Deferred源码

    前言: 如果你对jQuery.Callback回调对象不了解,或者只掌握其方法,但是没有通过阅读源码理解,可以先阅读 前一章jQuery.Callbacks源码解读二,因为只有完全理解jQuery.C ...

  3. javascript源代码学习之五——jQuery.deferred

    jQuery.Defered——异步队列用于管理一组回调函数(成功resolve,失败reject,消息progress),基于上一节实现的jQuery.callbacks完成. done,fail, ...

  4. 通过 ES6 Promise 和 jQuery Deferred 的异同学习 Promise

    Deferred 和 Promise ES6 和 jQuery 都有 Deffered 和 Promise,但是略有不同.不过它们的作用可以简单的用两句话来描述 Deffered 触发 resolve ...

  5. jquery.Deferred promise解决异步回调

    我们先来看一下编写AJAX编码经常遇到的几个问题: 1.由于AJAX是异步的,所有依赖AJAX返回结果的代码必需写在AJAX回调函数中.这就不可避免地形成了嵌套,ajax等异步操作越多,嵌套层次就会越 ...

  6. 第三十四课:jQuery Deferred详解2

    上一课主要分析了jQuery1.51版本的jQuery Deferred.在jQuery1.6中,jQuery Deferred添加了两个方法,always,pipe. always用来添加回调,无论 ...

  7. 第三十三课:jQuery Deferred详解1

    之前我们讲了Mochikit Deferred,JSDeferred,现在讲jQuery Deferred.首先,我们先来讲下他们的区别: 在保存回调函数时,Mochikit Deferred(doj ...

  8. 使用 jQuery Deferred 和 Promise 创建响应式应用程序

    这篇文章,我们一起探索一下 JavaScript 中的 Deferred 和 Promise 的概念,它们是 JavaScript 工具包(如Dojo和MochiKit)中非常重要的一个功能,最近也首 ...

  9. 利用 Jquery Deferred 异步你的程序

    最近在做公司QA系统改造时,有这样的一个场景. QA系统中有些数据项需要从JIRA平台(一个国外项目与事务跟踪工具)中获取,JIRA平台提供了很完善的Rest API. 现在的要求是,在QA系统中提交 ...

随机推荐

  1. pxe网络安装操作系统 原理与详细过程

    摘要:在实际工作中,我们经常会遇到这样的情况:想要安装Linux但是计算机不带光驱或软驱,或者是笔记本配置的非标准的软驱和光驱,如1394接口,USB接口等,在Linux安装时所引导的Linux内核一 ...

  2. Tour(KM算法)

    Tour Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others) Total Submi ...

  3. 驱动:中断【2】中断处理程序、中断上下文中处理延时及一些函数的调用规则(调IIC中断驱动有感)

    中断处理程序.中断上下文中处理延时及一些函数的调用规则(调IIC中断驱动有感)http://blog.csdn.net/samantha_sun/article/details/6790492 1,中 ...

  4. php上传文件,创建递归目录

    <?php $uid=$_REQUEST['uid']; $avatar = 'D:/avic/discuz/uc_server/data/avatar/'.get_avatar($uid, $ ...

  5. Android学习笔记:如何高效显示图片,避免内存溢出 和 ImageView无法显示大尺寸的图片

    因为手机的内存资源是有限的,每个app可使用的内存是受限的.而现在采用高分辨率拍的照片往往很大.如果加载时不注意方法,很有可能会引起java.lang.OutofMemoryError: bitmap ...

  6. 【HTTP 2】 序言

    笔者序 HTTP 2 这个关键词在各大 IT 网站出现频率节节升高,然而出现在大家视野里的,仍以浅析居多.虽有一部分实战(如 InfoQ 之前推送的 iOS App 网络层次架构优化),但针对 HTT ...

  7. KINGSO介绍

    kingso_intro - Taocode KINGSO介绍 KINGSO是一种高效的垂直化的搜索引擎,其包含query解析.检索.过滤.统计.排序功能,不包含抓取部分.它对商品搜索做了针对性的优化 ...

  8. 3DShader之投影贴图(Projective Texturing)

    相信大家都应该玩过CS或者CF吧,游戏里面有个喷图功能,就是按一个T键就能在墙上或者地板上喷出自己预先设定的图案. 而刚好这就是我们这个Shader所需实现的内容.由于没有潜伏者的贴图,我只有从这个图 ...

  9. 一步一步重写 CodeIgniter 框架 (5) —— 实现Controller,并加载Model

    CodeIgniter 框架采用MVC模式,而MVC模式中起纽带作用的就是C(控制器),在控制器的中通过加载模型获得数据,将数据传到视图中进行展示.本课将实现在控制器中加载模型. 1. 控制器的实现 ...

  10. iOS9适配系列教程

    链接地址:http://www.open-open.com/lib/view/open1443194127763.html 中文快速导航: iOS9网络适配_ATS:改用更安全的HTTPS(见Demo ...