創(chuàng)建可重載的方法
obj是被創(chuàng)建重載對(duì)象的方法
name是重載的函數(shù)名
fn是接受不同參數(shù)個(gè)數(shù)的函數(shù)體
實(shí)際上obj[name]并不是指定的fn,而是如下的匿名函數(shù),每一次為obj[name]
指定參數(shù)時(shí),就相當(dāng)于在匿名函數(shù)的后面加上了(arguments)使其立即執(zhí)行。
此時(shí)就可以通過(guò)匿名函數(shù)內(nèi)部的判斷條件來(lái)判斷是否有符合在重載函數(shù)內(nèi)的
arguments(通過(guò)閉包)來(lái)執(zhí)行函數(shù)
通過(guò)逐步的addMethod形成了類(lèi)似于鏈?zhǔn)秸{(diào)用的感覺(jué),
比如,第一次調(diào)用addMethod,addMethod中的obj[name]是一個(gè)保存有undefined的old和傳入的第一個(gè)函數(shù)體引用的匿名函數(shù),而第二次調(diào)用addMethod,首先會(huì)發(fā)現(xiàn)存在obj[name] (第一次定義的匿名函數(shù))(并且內(nèi)部還包含兩個(gè)引用),于是將這個(gè)賦給old,于是新old[name]就又成為了一個(gè)新的包含old和fn引用的匿名函數(shù),而這個(gè)old還包含了一個(gè)old和一個(gè)fn的引用一直延續(xù)下去
于是在調(diào)用obj[name] (后接著一些參數(shù))時(shí),就會(huì)先去找最新的那個(gè)obj[name],進(jìn)行判斷,若不對(duì),則通過(guò)old進(jìn)入上一級(jí)鏈?zhǔn)秸{(diào)用,去尋找上一層的obj[name]直到找的符合判斷條件的為止
function addMethod(obj,name,fn) {
//obj可能原本存在同名函數(shù)
var old = obj[name];
//注意這個(gè)函數(shù)并沒(méi)有被執(zhí)行
obj[name] = function () {
//在匿名函數(shù)內(nèi)保存著對(duì)導(dǎo)入的函數(shù)體fn的引用,在之后的調(diào)用中
//才可以取到相應(yīng)fn的值
if(fn.length===arguments.length) {
//實(shí)際上是使用當(dāng)前的對(duì)象傳入的參數(shù)來(lái)送入fn來(lái)執(zhí)行
return fn.apply(this, arguments);
}
//同理存在著對(duì)old的引用,old并未消亡
else if(typeof old === "function") {
return old.apply(this,arguments);
}
};
}