国产99久久精品_欧美日本韩国一区二区_激情小说综合网_欧美一级二级视频_午夜av电影_日本久久精品视频

最新文章專(zhuān)題視頻專(zhuān)題問(wèn)答1問(wèn)答10問(wèn)答100問(wèn)答1000問(wèn)答2000關(guān)鍵字專(zhuān)題1關(guān)鍵字專(zhuān)題50關(guān)鍵字專(zhuān)題500關(guān)鍵字專(zhuān)題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關(guān)鍵字專(zhuān)題關(guān)鍵字專(zhuān)題tag2tag3文章專(zhuān)題文章專(zhuān)題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專(zhuān)題3
問(wèn)答文章1 問(wèn)答文章501 問(wèn)答文章1001 問(wèn)答文章1501 問(wèn)答文章2001 問(wèn)答文章2501 問(wèn)答文章3001 問(wèn)答文章3501 問(wèn)答文章4001 問(wèn)答文章4501 問(wèn)答文章5001 問(wèn)答文章5501 問(wèn)答文章6001 問(wèn)答文章6501 問(wèn)答文章7001 問(wèn)答文章7501 問(wèn)答文章8001 問(wèn)答文章8501 問(wèn)答文章9001 問(wèn)答文章9501
當(dāng)前位置: 首頁(yè) - 科技 - 知識(shí)百科 - 正文

js更優(yōu)雅的兼容_javascript技巧

來(lái)源:懂視網(wǎng) 責(zé)編:小采 時(shí)間:2020-11-27 20:54:34
文檔

js更優(yōu)雅的兼容_javascript技巧

js更優(yōu)雅的兼容_javascript技巧:問(wèn)題種種做底層接口兼容,無(wú)非就是利用if,判斷客戶(hù)端支持哪個(gè)接口的問(wèn)題。最著名的例子就是事件: 代碼如下: var addEvent = function(e, what, how) { if (e.addEventListener) e.addEventListener(what, how, false)
推薦度:
導(dǎo)讀js更優(yōu)雅的兼容_javascript技巧:問(wèn)題種種做底層接口兼容,無(wú)非就是利用if,判斷客戶(hù)端支持哪個(gè)接口的問(wèn)題。最著名的例子就是事件: 代碼如下: var addEvent = function(e, what, how) { if (e.addEventListener) e.addEventListener(what, how, false)

問(wèn)題種種做底層接口兼容,無(wú)非就是利用if,判斷客戶(hù)端支持哪個(gè)接口的問(wèn)題。最著名的例子就是事件:
代碼如下:
var addEvent = function(e, what, how) {
if (e.addEventListener) e.addEventListener(what, how, false)
else if (e.attachEvent) e.attachEvent('on' + what, how)
}

這里考慮了給元素綁定事件時(shí)可能遇到的兩種狀況——標(biāo)準(zhǔn)的W3C DOM接口以及DHTML提供的接口。當(dāng)然這個(gè)例子還很粗糙,但足夠說(shuō)明問(wèn)題了。

原先的方法是在兼容層調(diào)用有現(xiàn)場(chǎng)判斷并進(jìn)入相應(yīng)的if分支。很顯然,這種“現(xiàn)場(chǎng)判斷”的方法效率并不高。后來(lái),人們采用這樣的辦法:
代碼如下:
if (MSIE) {
addEvent = function(e, what, how) {
e.attachEvent('on' + what, how);
}
} else {
addEvent = function(e, what, how) {
e.addEventListener(what, how);
}
}

在一次判斷后給addEvent綁定不同的代碼,從而免去了運(yùn)行時(shí)的分支判斷。

很可惜,這個(gè)問(wèn)題也不小。首先把“采用attachEvent”和“客戶(hù)端是MSIE”綁定在一起是個(gè)很過(guò)時(shí)的想法。假如微軟哪天良心發(fā)現(xiàn)了怎么辦?這事情現(xiàn)在就發(fā)生了——IE9明確支持了DOM接口,甚至DOM3都支持。結(jié)果,就這個(gè)“良心發(fā)現(xiàn)”的舉動(dòng)會(huì)毀掉許多前端庫(kù),他們必須被迫修改代碼(如同IE8來(lái)時(shí)那樣)。況且這種做法沒(méi)有考慮“未知的客戶(hù)端”——據(jù)我所知,Google發(fā)布Chrome后也導(dǎo)致不少類(lèi)庫(kù)重寫(xiě)代碼。

特性檢測(cè)那究竟該怎么做?特性檢測(cè)就可以最大限度地避免“新客戶(hù)端”帶來(lái)的麻煩——通過(guò)一組在類(lèi)庫(kù)初始化時(shí)定義的代碼來(lái)檢測(cè)客戶(hù)端擁有的特性,并利用這一組檢測(cè)值綁定類(lèi)庫(kù)代碼:
代碼如下:
var supportsAddEventListener = !!(checkerElement.addEventListener);
if (supportsAddEventListener) {
addEvent = function(e, what, how) {
e.addEventListener(what, how);
}
} else if (supportsAttachEvent) {
addEvent = function(e, what, how) {
e.attachEvent('on' + what, how);
}
}

特性檢測(cè)實(shí)際上是將“使用某個(gè)客戶(hù)端”和“支持某個(gè)特性”進(jìn)行解耦——讓if分支直接針對(duì)“特性有無(wú)”(接口是否一致)判斷,從而消除客戶(hù)端制造商“良心發(fā)現(xiàn)”造成的“好心辦壞事”。事實(shí)上這么做也是符合歷史潮流之選——當(dāng)標(biāo)準(zhǔn)接口逐漸普及,客戶(hù)端之間漸漸“表征一致”時(shí),為什么不做個(gè)一致的兼容層接口呢?

跌落讓我們重新看看這些代碼。通常,一條利用特性檢測(cè)進(jìn)行兼容的代碼往往是這樣:
代碼如下:
if (new_interface_detected) {
comp = function() {uses_new_interface};
} else if (old_interface_detected) {
comp = function() {uses_old_interface};
} else {
throw new Error('Unadaptable!')
}

換言之,過(guò)程是:

如果客戶(hù)端支持新接口,就將兼容層綁定到新接口上
否則,如果客戶(hù)端支持老接口/不一致接口,就將兼容層綁定到老接口上
否則,如果可以的話,給出錯(cuò)誤回饋
亦即,兼容層程序是從高空“掉”下來(lái),如果客戶(hù)端支持“高級(jí)”特性(新接口、標(biāo)準(zhǔn)接口)就將它“接住”——兼容層就有了歸宿;否則繼續(xù)向下掉——哦,老接口接住了,就用老接口;如果一直沒(méi)人接住,于是——啪——摔倒了地上,并且用最后一口氣喊一聲:“你用的客戶(hù)端太小眾,我拿你沒(méi)辦法了!”

這和什么比較像?事實(shí)上,如果你了解JavaScript對(duì)象系統(tǒng)的機(jī)理,你就可以類(lèi)比:這不就是原型嘛!原型系統(tǒng)就是利用了這種跌落——尋找某個(gè)成員,如果它在這個(gè)對(duì)象里定義了,就返回之;否則沿著原型鏈向上搜(沒(méi)錯(cuò),這次是向上的),如此重復(fù),直到真的連原型鏈都到頭的時(shí)候,返回個(gè)undefined。

說(shuō)做就做!這里同樣用addEvent為例。首先,我們定義一個(gè)空驅(qū)動(dòng),它里面什么都不包含:

var nullDriver = {}然后,就是創(chuàng)建個(gè)對(duì)象,并且把原型鏈指向它。在ECMA V5時(shí)代,我們可以用Object.create,可惜,現(xiàn)在還有N多老客戶(hù)端(否則做什么兼容啊),所以自己craft個(gè)函數(shù):
代碼如下:
var derive = Object.create ? Object.create: function() {
var T = function() {};
return function(obj) {
T.prototype = obj;
return new T
}
}()

這個(gè)用法你可能會(huì)覺(jué)得很詭異,但它工作起來(lái)一點(diǎn)問(wèn)題沒(méi)有,速度也不慢——能達(dá)到Object.create的一半。我們就用這個(gè)derive開(kāi)動(dòng):
代碼如下:
var dhtmlDriver = derive(nullDriver);
var dhtmlDriverBugfix = derive(dhtmlDriver);這里的bugfix是針對(duì)一些“bug”和特殊情況定義的特別Driver。這里你可以忽略它。好了,DHTML里面addEvent是什么來(lái)著?

if (supportsAttachEvent) {
dhtmlDriver.addEvent = function(e, what, how) {
e.attachEvent('on' + what, how)
}
}

然后呢?位于原型鏈最前端的應(yīng)該是W3C的標(biāo)準(zhǔn)驅(qū)動(dòng)啊,寫(xiě)上!
代碼如下:
var w3cDriver = derive(dhtmlDriverBugfix);
var w3cDriverBugfix = derive(w3cDriver);

if (supportsAddEventListener) {
w3cDriver.addEvent = function(e, what, how) {
e.addEventListener(what, how)
}
}

最后,我們就放個(gè)東西上去做最后調(diào)用的接口。(因?yàn)閣3cDriverBugfix太難看……)
代碼如下:
var driver = derive(w3cDriverBugfix);

然后就調(diào)用好了。看,這就讓那些長(zhǎng)得嚇人的分支判斷變得簡(jiǎn)單有效,但不失fallback本色:在支持addEventListener上調(diào)用addEvent等價(jià)于調(diào)用w3cDriver.addEvent,而在不支持addEventListener的客戶(hù)端上就會(huì)跌落到底下,比如調(diào)用dhtmlDriver.addEvent。另外,進(jìn)行bugfix也很容易——可以在專(zhuān)門(mén)的“bugfix”層進(jìn)行hook,而原有層絲毫不受影響。

等等,繼承這么多層會(huì)很慢么?誠(chéng)然,那么深的原型鏈肯定會(huì)慢,不過(guò)我有辦法。還記得給對(duì)象的屬性寫(xiě)入時(shí)會(huì)發(fā)生什么事情嗎?
代碼如下:
var ego = function(x) {return x}
for (var each in driver) {
if (! (each in nullDriver)) {
driver[each] = ego(driver[each])
}
}

沒(méi)錯(cuò),原來(lái)高企在原型鏈上面的方法會(huì)“嘩”的一下掉到最下面!這回不用沿著原型鏈向上搜了,直接從最底端獲取屬性即可。這里用ego函數(shù)的原因是防止一些瀏覽器“優(yōu)化掉”這里的代碼。

總結(jié)雖然這里談兼容,可是,它的精華卻在語(yǔ)言特性上——利用原型繼承,我們可以很優(yōu)雅地完成這個(gè)令人頭疼的操作。是的,框架的美感不應(yīng)該只在外表,其內(nèi)部——即使是最最令人煩的內(nèi)部——也同樣要優(yōu)雅。

這里的技術(shù)可以在dess中找到。

聲明:本網(wǎng)頁(yè)內(nèi)容旨在傳播知識(shí),若有侵權(quán)等問(wèn)題請(qǐng)及時(shí)與本網(wǎng)聯(lián)系,我們將在第一時(shí)間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文檔

js更優(yōu)雅的兼容_javascript技巧

js更優(yōu)雅的兼容_javascript技巧:問(wèn)題種種做底層接口兼容,無(wú)非就是利用if,判斷客戶(hù)端支持哪個(gè)接口的問(wèn)題。最著名的例子就是事件: 代碼如下: var addEvent = function(e, what, how) { if (e.addEventListener) e.addEventListener(what, how, false)
推薦度:
標(biāo)簽: 小技巧 js 支持
  • 熱門(mén)焦點(diǎn)

最新推薦

猜你喜歡

熱門(mén)推薦

專(zhuān)題
Top
主站蜘蛛池模板: 国产成人久久精品一区二区三区 | www.亚洲欧美 | 欧美伊香蕉久久综合类网站 | 中出在线播放 | www.久久99 | 一区二区中文字幕 | 欧美视频免费在线 | 日韩欧美高清 | 成人国产一区二区三区精品 | 精品在线免费播放 | 欧美在线一区二区 | 日本一区二区三区不卡在线视频 | 日韩阿v | 精品日韩欧美一区二区三区 | 殴美aⅴ| 欧美成人看片一区二区三区尤物 | 精品一区二区三区免费观看 | 日本不卡视频一区二区 | 国产麻豆a一级毛片爽爽影院 | 91成人爽a毛片一区二区 | 在线看欧美 | 欧美专区第一页 | 在线h片 | 在线观看免费精品国自产 | 免费看一级黄色毛片 | 精品一区二区三区四区电影 | 日韩黄页 | 成人国内精品久久久久影院 | 91精品成人免费国产 | 亚洲qvod图片区电影 | 91发布页 | 麻豆成人在线 | 国产精品亚洲一区二区三区久久 | 国产欧美一区二区精品性色99 | 欧美爱爱网址 | 久久99一区 | 国产不卡的一区二区三区四区 | 欧美日韩国内 | 国产码欧美日韩高清综合一区 | 国产欧美成人一区二区三区 | 国产日韩欧美亚洲综合首页 |