$.get( "test.php" ).always(function() {
alert( "$.get completed with success or error callback arguments" );
});
测试代码2:(deferred.done)
$.get( "test.php" ).done(function() {
alert( "$.get succeeded" );
});
function fn1() {
$( "p" ).append( " 1 " );
}
function fn2() {
$( "p" ).append( " 2 " );
}
function fn3( n ) {
$( "p" ).append( n + " 3 " + n );
}
//创建deferred对象
var dfd = $.Deferred();
dfd.done().done( [ fn1, fn2 ], fn3, [ fn2, fn1 ] ).done(function( n ) {
$( "p" ).append( n + " we're done." );
});
//点击按钮时候Deferred调用resolve方法!打印结果[1 2 and 3 and 2 1 and we're done.]
$( "button" ).on( "click", function() {
dfd.resolve( "and" );
});
测试代码3:(deferred.fail)
var defer = $.Deferred(),
//pipe方法
filtered = defer.pipe(function( value ) {
return value * 2;
});
//传入参数5
defer.resolve( 5 );
filtered.done(function( value ) {
alert( "Value is ( 2*5 = ) 10: " + value );
});
//过滤reject值
var defer = $.Deferred(),
//传入两个参数,第二个是failFilter
filtered = defer.pipe( null, function( value ) {
return value * 3;
});
defer.reject( 6 );
//这里是filtered.fail事件,上面是done事件
filtered.fail(function( value ) {
alert( "Value is ( 3*6 = ) 18: " + value );
});
//链式任务
var request = $.ajax( url, { dataType: "json" } ),
chained = request.pipe(function( data ) {
return $.ajax( url2, { data: { user: data.userId } } );
});
chained.done(function( data ) {
// data retrieved from url2 as provided by the first request
});
测试代码9(deferred.progress( progressCallbacks [, progressCallbacks ] )产生progress事件后回调函数)
function asyncEvent() {
var dfd = jQuery.Deferred();
// Resolve after a random interval
setTimeout(function() {
dfd.resolve( "hurray" );
}, Math.floor( 400 + Math.random() * 2000 ) );
// Reject after a random interval
setTimeout(function() {
dfd.reject( "sorry" );
}, Math.floor( 400 + Math.random() * 2000 ) );
// Show a "working..." message every half-second
setTimeout(function working() {
if ( dfd.state() === "pending" ) {
alert("pending");
dfd.notify( "working... " );
setTimeout( working, 500 );
}
}, 1 );
// Return the Promise so caller can't change the Deferred
//返回一个Promise,因此调用者无法在外面改变Deferred对象的状态!
return dfd.promise();
}
// Attach a done, fail, and progress handler for the asyncEvent
//为异步事件添加done,fail,progress事件!
$.when( asyncEvent() ).then(
function( status ) {
alert( status + ", things are going well" );
},
function( status ) {
alert( status + ", you fail this time" );
},
function( status ) {
$( "body" ).append( status );
}
);
//已经存在的obj对象
var obj = {
hello: function( name ) {
alert( "Hello " + name );
}
},
//创建Deferred对象
defer = $.Deferred();
// 使得obj具有promise的方法
defer.promise( obj );
//改变状态为resolved
defer.resolve( "John" );
//将obj当作Promise使用
obj.done(function( name ) {
obj.hello( name ); //"Hello John"。将resolve的参数传入done函数
}).hello( "Karl" ); //"Hello Karl"。该hello方法是这个promise独有的方法
测试代码11:(deferred.reject)
var filterResolve = function() {
var defer = $.Deferred(),
filtered = defer.then(function( value ) {
return value * 2;
});
defer.resolve( 5 );
filtered.done(function( value ) {
$( "p" ).html( "Value is ( 2*5 = ) 10: " + value );//打印10
});
};
//过滤reject
var defer = $.Deferred(),
filtered = defer.then( null, function( value ) {//fail
return value * 3;
});
defer.reject( 6 );
filtered.fail(function( value ) {
alert( "Value is ( 3*6 = ) 18: " + value );
});
//链式调用
var request = $.ajax( url, { dataType: "json" } ),
chained = request.then(function( data ) {
return $.ajax( url2, { data: { user: data.userId } } );
});
chained.done(function( data ) {
// data retrieved from url2 as provided by the first request
});
测试代码17:(jQuery.Deferred( [beforeStart ] ))var div = $( "" );
div.promise().done(function( arg1 ) {
// 马上调用,打印"true"
alert( this === div && arg1 === div );
});
//当所有的动画完成以后,返回的Promise变成resolved状态,包括那些一开始就存在于callback集合中的和那些后来添加的
$( "button" ).on( "click", function() {
$( "p" ).append( "Started..." );
$( "div" ).each(function( i ) {
$( this ).fadeIn().fadeOut( 1000 * ( i + 1 ) );
});
$( "div" ).promise().done(function() {
$( "p" ).append( " Finished! " );
});
});
//用$.when()将返回的Pormise对象变成resolved,.promise可以用于jQuery集合
var effect = function() {
return $( "div" ).fadeIn( 800 ).delay( 1200 ).fadeOut();
};
$( "button" ).on( "click", function() {
$( "p" ).append( " Started... " );
$.when( effect() ).done(function() {
$( "p" ).append( " Finished! " );
});
});
下面我们开始jQuery是怎么完成的,请提前阅读Callbacks相关问题
源码问题1:测试每一个Deferred对应的三个Callbacks作用
var dfd=$.Deferred();
setInterval(function()
{
alert(111);
dfd.resolve();
},1000)
dfd.done(function(){
alert("成功");
}).fail(function()
{
alert("失败");
}).progress(function(){alert("进行中!")})
note:这个例子done中的函数只会执行一次,以后在调用resolve的时候不会触发done方法,reject也是一样!但是如果把上面修改为notify那么一直会交替弹出:[111,进行中]。为什么完成和未完成只能触发一次,而进行中要连续触发?因为成功或者失败是一种状态,然而进行中可以连续触发,这对进度条等有作用!之所以能够实现只会调用一次,因为源码中用的是jQuery.Callbacks("once memory"),
因为once+memory的组合调用了一次就会把list=[]所以下次继续resolve(底层调用fireWith),list已经为空了!没有效果!
源码问题2:(memory在这里有什么作用,复习Callbacks内容)
var cb=$.Callbacks("memory");
cb.add(function(){
alert(1);
});
cb.fire();//点击按钮以后会把新的函数添加入cb中间去,而memory表示不需要自己调用就能执行
//因为add里面会自己执行!会拿着上一次的参数执行,上一次如果没有参数那么就是undefined!
$("input").click(function()
{
cb.add(function()
{
alert(2);
})
});
note:到了这里你应该有了想法了,memory如果没有手动调用的时候会自己调用!那么在上面的例子中,如果我们已经resolve了,也就是list集合中的函数已经全部调用了,那么我们现在重新添加一个done方法会怎么样呢?
立即执行!之所以会这样就是memory记住了上一次的参数,不用手动调用而会立即调用新的函数!(这是为什么我总结:如果没有手动调用,在有memory的情况的下只会用老参数调用新函数;如果手动调用的时候才会发生两个阶段,阶段一就是用老参数调用新函数,阶段二就是用新函数调用所有的函数)这是为什么resolve以后再次添加done方法这个方法会立即执行的原因!这和下面的例子是一样的!
源码问题3:
var dfd=$.Deferred();
setTimeout(function()
{
alert(111);
dfd.resolve();
},1000)
dfd.done(function(){
alert("aaa");
}).fail(function()
{
alert("失败");
}).progress(function(){alert("进行中!")})
$("input").click(function()
{
cb.add(function()
{
alert("bbb");
})
});
//点击的时候状态已经完成了,也就是已经resolve了,当点击按钮的时候马上弹出bbb!只要状态已经完成,那么你再进行done操作的时候会立即执行,
//这就是memory起的作用!
源码问题4:(简单执行)
function aaa()
{
var dfd=$.Deferred();
setTimeout(function()
{
dfd.resolve();
},1000)
}
var newDfd=aaa();
newDfd.done(function(){alert("成功")}).fail(function(){alert("失败");})
// 很显然会弹出"成功",因为resolve会调用底层的fireWith,fireWith就会把集合中的函数全部执行,而resolve对应的Callbacks就是done!
测试问题4:(Deferred对象的状态很容易被修改!)
function aaa()
{
var dfd=$.Deferred();
setTimeout(function()
{
dfd.resolve();
},1000)
return dfd;
}
var newDfd=aaa();
newDfd.done(function(){alert("成功")}).fail(function(){alert("失败");})
newDfd.reject();//上面的定时器还没有执行,失败已经触发了,这时候成功不会调用!说明状态很容易修改掉
测试问题5:(为什么源码中的要封装promise对象,该对象没有resolve,resolveWith等修改状态的方法)
function aaa()
{
var dfd=$.Deferred();
setTimeout(function()
{
dfd.resolve();
},1000)
return dfd.promise();
}
var newDfd=aaa();
newDfd.done(function(){alert("成功")}).fail(function(){alert("失败");})
newDfd.reject();//这时候不能修改状态,还会报错,因为promise对象没有reject!
测试问题6:(pipe方法的使用)
var dfd=$.Deferred();
setTimeout(function()
{
dfd.resolve("hi");
},1000)
//返回一个新的延迟对象!返回值作为延迟对象的resove的参数
var newDfd= dfd.pipe(function(){
return arguments[0]+"xxx"
})
//这个会立即执行,因为上面已经resolve了!
newDfd.done(function(){
alert(arguments[0]);
});
jQuery.Deferred对象源码:
jQuery.extend({
Deferred: function( func ) {
var tuples = [
// action, add listener, listener list, final state
[ "resolve", "done", jQuery.Callbacks("once memory"), "resolved" ],
[ "reject", "fail", jQuery.Callbacks("once memory"), "rejected" ],
[ "notify", "progress", jQuery.Callbacks("memory") ]
],
state = "pending",
promise = {
state: function() {
return state;
},
//不管成功还是失败都是会回调的!也就是把这个函数同时添加如done对应的Callbacks对象中也添加进fail对于的Callbacks对象中
//记住这里的done,fail对应于相应的Callbacks的add方法!
always: function() {
deferred.done( arguments ).fail( arguments );
return this;
},
//分开写,第一个是完成的回调,第二个是失败,第三个是进度
then: function( /* fnDone, fnFail, fnProgress */ ) {
var fns = arguments;
//pipe方法和then方式是同一个方法,所以返回值是一个Defered类型,参见博客中“测试问题6”
//jQuery.Deferred中传入一个函数表示立即执行!
return jQuery.Deferred(function( newDefer ) {
jQuery.each( tuples, function( i, tuple ) {
//假如传入了三个函数,那么第一次i为0,获取到第一个函数也就是成功回调函数
var fn = jQuery.isFunction( fns[ i ] ) && fns[ i ];
// deferred[ done | fail | progress ] for forwarding actions to newDefer
//第一次调用done函数,第二次是fail,第三次是notify函数
//deferred[ tuple[1] ]就是deferred["done"]这就是add方法,也就是把后面的那个函数添加到done里面去执行!
deferred[ tuple[1] ](function() {
//获取第一个成功回调函数的返回值,这里的arguments就是传递给resolve或者reject或者notify的参数!
//之所以arguments是resolve的参数是因为:这里直接在done的Callbacks列表中加入了这个匿名函数,
//如果外面已经resolve了那么这里就会立即执行!
var returned = fn && fn.apply( this, arguments );
//如果返回值是promise的处理逻辑
if ( returned && jQuery.isFunction( returned.promise ) ) {
returned.promise()
.done( newDefer.resolve )
.fail( newDefer.reject )
.progress( newDefer.notify );
} else {
//否则:
newDefer[ tuple[ 0 ] + "With" ]( this === promise ? newDefer.promise() : this, fn ? [ returned ] : arguments );
}
});
});
fns = null;
}).promise();
},
// Get a promise for this deferred
// If obj is provided, the promise aspect is added to the object
//如果没有参数那么返回Deferred对象对应的promise,如果有参数那么就是让参数继承promise所有的方法
//后面用到这里让Deferred继承promise的所有的方法!
promise: function( obj ) {
return obj != null ? jQuery.extend( obj, promise ) : promise;
}
},
deferred = {};
// Keep pipe for back-compat
promise.pipe = promise.then;
// Add list-specific methods
jQuery.each( tuples, function( i, tuple ) {
var list = tuple[ 2 ],
stateString = tuple[ 3 ];
//promise里面封装了三个Callbacks对象,promise的done|faile|progress对应于相应对象的add方法
//done里面的函数会被封装到jQuery.Callbacks("once memory")这个Callbacks对象上面,
//fail会被封装到jQuery.Callbacks("once memory")这个Callbacks对象上面,
//progress会被封装到jQuery.Callbacks("memory")这个Callbacks对象上面,而且这三个Callbacks是互不影响的!
//也就是add时候只是添加到特定的Callbacks里面!
// promise[ done | fail | progress ] = list.add
promise[ tuple[1] ] = list.add;
// Handle state
//触发了未完成那么不能触发完成,触发了完成不可能触发为完成的代码,这里就是用来控制的!
//第一次是resolve,所以tupes[i^1]就是reject,也就是resolve执行了不能再执行reject,即让所有的reject全部调用disable!
//这里要深入理解Callbacks才行,原理为:
//(1)如果是resolve类型那么我把这个reject中的集合jQuery.Callbacks("once memory")全部调用disable
//也就说这个集合里面的list=stack=memory=undefined!(请仔细阅读Callbacks源码)
//(2)而且这个函数是放在最前面的,也就是每一次开始调用的时候他都是最前调用的,以防有人调用reject改变状态!
if ( stateString ) {
list.add(function() {
// state = [ resolved | rejected ]
state = stateString;
// [ reject_list | resolve_list ].disable; progress_list.lock
}, tuples[ i ^ 1 ][ 2 ].disable, tuples[ 2 ][ 2 ].lock );
}
// deferred[ resolve | reject | notify ]
deferred[ tuple[0] ] = function() {
deferred[ tuple[0] + "With" ]( this === deferred ? promise : this, arguments );
return this;
};
//(1)Deferred对象的resolveWidth等方法都是对应于Callbacks对象的fireWith方法!
//而且Deferred对象的resolve,reject,notify方法调用的是resolveWith,rejectWith,notifyWidth
//也就是归根到底,resolve方法还是调用fireWidth只是要传进去相应的参数而已!
//从上面的代码可以看到,如果调用resolveWith的调用者是Deferred对象,那么调用fireWith时候传入的第一个参数就是promise
//如果调用resolveWidth的不是Deferred对象,那么调用fireWidth时候的第一个参数就是调用者,第二个参数是resolveWith传入的参数!
//(2)这里也就是说对于resolveWith来说,他对应于jQuery.Callbacks("once memory")的fireWith也就说resolve调用了只会调用这里面的函数的队列
//而这里面函数的队列对应于就是done中间的队列,done中间的函数队列又对应于jQuery.Callbacks("once memory")中add的方法!
deferred[ tuple[0] + "With" ] = list.fireWith;
});
// Make the deferred a promise
promise.promise( deferred );
// Call given func if any
if ( func ) {
func.call( deferred, deferred );
}
// All done!
return deferred;
}
总结:
(1)Deferred对象有三个方法done,fail,notify,他们对应于不同的Callbacks对象的add方法,所以我们在这三个方法里面直接添加函数!如果是done方法相当于直接在done对于的Callbacks中添加了回调函数,构建出一个list集合!那么如何触发这个集合中的函数呢?Deferred对象专门为这个Callbacks准备了resolve,resolveWith等方法(底层调用该Callbacks的fireWith方法),该方法相当于执行通过done方法添加的Callbacks中的所有的函数!(其它函数也是同样的思路!)
(2)回调函数在promise方法上面,但是状态改变的方法全部在deferred对象上!Deferred只是多了修改状态的方法!
(3)如果调用resolveWith的调用者是Deferred对象,那么调用fireWith时候传入的第一个参数就是promise, 如果调用resolveWidth的不是Deferred对象,那么调用fireWidth时候的第一个参数就是调用者,第二个参数是resolveWith传入的参数!