不唐突的JavaScript的七條準(zhǔn)則整理收集第1/2頁(yè)
(JavaScript是一個(gè)不可靠的助手)
可能不唐突的JavaScript 的最重要的一個(gè)特性就是——你要停止任何假設(shè):
* 不要假設(shè)JavaScript是可用的,你最好認(rèn)為它很有可能是不可用的,而不是直接依賴于它。
* 在你經(jīng)過(guò)測(cè)試確認(rèn)一些方法和屬性可以使用之前,不要假設(shè)瀏覽器支持它們。
* 不要假設(shè)HTML代碼如你想象的那樣正確,每次都要進(jìn)行檢查,并且當(dāng)其不可用的時(shí)候就什么也不要做。
* 讓JavaScript的功能獨(dú)立于輸入設(shè)備
* 要記住其他的腳本可能會(huì)影響你的JavaScript的功能,所以要保證你的腳本的作用域盡可能地安全。
在開(kāi)始設(shè)計(jì)你的腳本之前,要考慮的第一件事情就是檢查一下你要為其編寫腳本的HTML代碼,看看有什么東西可以幫助你達(dá)到目的。
2.找出鉤子和節(jié)點(diǎn)關(guān)系
(HTML是腳本的基石)
在開(kāi)始編寫腳本之前,要先看一下你要為之編寫JavaScript的HTML。如果HTML是未經(jīng)組織的或者未知的,那么你幾乎不可能有一個(gè)好的腳本編寫方案——很可能就會(huì)出現(xiàn)下面的情況:要么是會(huì)用JavaScript創(chuàng)建太多標(biāo)記,要么就是應(yīng)用太依賴于JavaScript。
在HTML中有一些東西需要考慮,那就是鉤子和節(jié)點(diǎn)關(guān)系。
<1>.HTML 鉤子
HTML最初的和最重要的鉤子就是ID,而且ID可以通過(guò)最快的DOM方法——getElementById 訪問(wèn)到。如果在一個(gè)有效的HTML文檔中所有的ID都是獨(dú)一無(wú)二的話(在IE中關(guān)于name 和 ID 有一個(gè)bug,不過(guò)有些好的類庫(kù)解決了這個(gè)問(wèn)題),使用ID就是安全可靠的,并且易于測(cè)試。
其他一些鉤子就是是HTML元素和CSS類,HTML元素可以通過(guò)getElementsByTagName方法訪問(wèn),而在多數(shù)瀏覽器中都還不能通過(guò)原生的DOM方法來(lái)訪問(wèn)CSS類。不過(guò),有很多外部類庫(kù)提供了可以訪問(wèn)CSS類名(類似于 getElementsByClassName) 的方法。
<2>.HTML 節(jié)點(diǎn)關(guān)系
關(guān)于HTML的另外比較有意思的一點(diǎn)就是標(biāo)記之間的關(guān)系,思考下面的問(wèn)題:
* 要怎樣才可以最容易地、通過(guò)最少的DOM遍歷來(lái)到達(dá)目標(biāo)節(jié)點(diǎn)?
* 通過(guò)修改什么標(biāo)記,可以盡可能多地訪問(wèn)到需要修改的子節(jié)點(diǎn)?
* 一個(gè)給定的元素有什么屬性或信息可以用來(lái)到達(dá)另外一個(gè)元素?
遍歷DOM很耗資源而且速度很慢,這就是為什么要盡量使用瀏覽器中已經(jīng)在使用的技術(shù)來(lái)做這件事情。
3.把遍歷交給專家來(lái)做
(CSS,更快地遍歷DOM)
有關(guān)DOM的腳本和使用方法或?qū)傩裕╣etElementsByTagName, nextSibling, previousSibling, parentNode以及其它)來(lái)遍歷DOM似乎迷惑了很多人,這點(diǎn)很有意思。而有趣的是,我們其實(shí)早已經(jīng)通過(guò)另外一種技術(shù)—— CSS ——做了這些事情。
CSS 是這樣一種技術(shù),它使用CSS選擇器,通過(guò)遍歷DOM來(lái)訪問(wèn)目標(biāo)元素并改變它們的視覺(jué)屬性。一段復(fù)雜的使用DOM的JavaScript可以用一個(gè)CSS選擇器取代:
var n = document.getElementById('nav');
if(n){
var as = n.getElementsByTagName('a');
if(as.length > 0){
for(var i=0;as[i];i++){
as[i].style.color = ‘#369′;
as[i].style.textDecoration = ‘none';
}
}
}
/* 下面的代碼與上面功能一樣 */
#nav a{
color:#369;
text-decoration:none;
}
這是一個(gè)可以好好利用的很強(qiáng)大的技巧。你可以通過(guò)動(dòng)態(tài)為DOM中高層的元素添加class 或者更改元素ID來(lái)實(shí)現(xiàn)這一點(diǎn)。如果你使用DOM為文檔的body添加了一個(gè)CSS類,那么設(shè)計(jì)師就很可以容易地定義文檔的靜態(tài)版本和動(dòng)態(tài)版本。
JavaScript:
var dynamicClass = 'js';
var b = document.body;
b.className = b.className ? b.className + ' js' : 'js';
CSS:
/* 靜態(tài)版本 */
#nav {
....
}
/* 動(dòng)態(tài)版本 */
body.js #nav {
....
}
4.理解瀏覽器和用戶
(在既有的使用模式上創(chuàng)建你所需要的東西)
不唐突的JavaScript 中很重要的一部分就是理解瀏覽器是如何工作的(尤其是瀏覽器是如何崩潰的)以及用戶期望的是什么。不考慮瀏覽器你也可以很容易地使用JavaScript 創(chuàng)建一個(gè)完全不同的界面。拖拽界面,折疊區(qū)域,滾動(dòng)條和滑動(dòng)塊都可以使用JavaScript創(chuàng)建,但是這個(gè)問(wèn)題并不是個(gè)簡(jiǎn)單的技術(shù)問(wèn)題,你需要思考下面的問(wèn)題:
* 這個(gè)新界面可以獨(dú)立于輸入設(shè)備么?如果不能,那么可以依賴哪些東西?
* 我創(chuàng)建的這個(gè)新界面是否遵循了瀏覽器或者其它富界面的準(zhǔn)則(你可以通過(guò)鼠標(biāo)在多級(jí)菜單中直接切換嗎?還是需要使用tab鍵?)
* 我需要提供什么功能但是這個(gè)功能是依賴于JavaScript的?
最后一個(gè)問(wèn)題其實(shí)不是問(wèn)題,因?yàn)槿绻枰憔涂梢允褂肈OM來(lái)憑空創(chuàng)建HTML。關(guān)于這點(diǎn)的一個(gè)例子就是“打印”鏈接,由于瀏覽器沒(méi)有提供一個(gè)非 JavaScript的打印文檔功能,所以你需要使用DOM來(lái)創(chuàng)建這類鏈接。同樣地,一個(gè)實(shí)現(xiàn)了展開(kāi)和收縮內(nèi)容模塊的、可以點(diǎn)擊的標(biāo)題欄也屬于這種情況。標(biāo)題欄不能被鍵盤激活,但是鏈接可以。所以為了創(chuàng)建一個(gè)可以點(diǎn)擊的標(biāo)題欄你需要使用JavaScript將鏈接加入進(jìn)去,然后所有使用鍵盤的用戶就可以收縮和展開(kāi)內(nèi)容模塊了。
解決這類問(wèn)題的極好的資源就是設(shè)計(jì)模式庫(kù)。至于要知道瀏覽器中的哪些東西是獨(dú)立于輸入設(shè)備的,那就要靠經(jīng)驗(yàn)的積累了。首先你要理解的就是事件處理機(jī)制。
5.理解事件
(事件處理會(huì)引起改變)
事件處理是走向不唐突的JavaScript的第二步。重點(diǎn)不是讓所有的東西都變得可以拖拽、可以點(diǎn)擊或者為它們添加內(nèi)聯(lián)處理,而是理解事件處理是一個(gè)可以完全分離出來(lái)的東西。我們已經(jīng)將HTML,CSS和JavaScript分離開(kāi)來(lái),但是在事件處理的分離方面卻沒(méi)有走得很遠(yuǎn)。
事件處理器會(huì)監(jiān)聽(tīng)發(fā)生在文檔中元素上的變化,如果有事件發(fā)生,處理器就會(huì)找到一個(gè)很奇妙的對(duì)象(一般會(huì)是一個(gè)名為e的參數(shù)),這個(gè)對(duì)象會(huì)告訴元素發(fā)生了什么以及可以用它做什么。
對(duì)于大多數(shù)事件處理來(lái)說(shuō),真正有趣的是它不止發(fā)生在你想要訪問(wèn)的元素上,還會(huì)在DOM中較高層級(jí)的所有元素上發(fā)生(但是并不是所有的事件都是這樣,focus和blur事件是例外)。舉例來(lái)說(shuō),利用這個(gè)特性你可以為一個(gè)導(dǎo)航列表只添加一個(gè)事件處理器,并且使用事件處理器的方法來(lái)獲取真正觸發(fā)事件的元素。這種技術(shù)叫做事件委托,它有幾點(diǎn)好處:
* 你只需要檢查一個(gè)元素是否存在,而不需要檢查每個(gè)元素
* 你可以動(dòng)態(tài)地添加或者刪除子節(jié)點(diǎn)而并不需要?jiǎng)h除相應(yīng)的事件處理器
* 你可以在不同的元素上對(duì)相同的事件做出響應(yīng)
需要記住的另一件事是,在事件向父元素傳播的時(shí)候你可以停止它而且你可以覆寫掉HTML元素(比如鏈接)的缺省行為。不過(guò),有時(shí)候這并不是個(gè)好主意,因?yàn)闉g覽器賦予HTML元素那些行為是有原因的。舉個(gè)例子,鏈接可能會(huì)指向頁(yè)面內(nèi)的某個(gè)目標(biāo),不去修改它們能確保用戶可以將頁(yè)面當(dāng)前的腳本狀態(tài)也加入書(shū)簽。
6.為他人著想
(命名空間,作用域和模式)
你的代碼幾乎從來(lái)不會(huì)是文檔中的唯一的腳本代碼。所以保證你的代碼里沒(méi)有其它腳本可以覆蓋的全局函數(shù)或者全局變量就顯得尤為重要。有一些可用的模式可以來(lái)避免這個(gè)問(wèn)題,最基礎(chǔ)的一點(diǎn)就是要使用 var 關(guān)鍵字來(lái)初始化所有的變量。假設(shè)我們編寫了下面的腳本:
var nav = document.getElementById('nav');
function init(){
// do stuff
}
function show(){
// do stuff
}
function reset(){
// do stuff
}
上面的代碼中包含了一個(gè)叫做nav的全局變量和名字分別為 init,show 和 reset 的三個(gè)函數(shù)。這些函數(shù)都可以訪問(wèn)到nav這個(gè)變量并且可以通過(guò)函數(shù)名互相訪問(wèn):
var nav = document.getElementById('nav');
function init(){
show();
if(nav.className === 'show'){
reset();
}
// do stuff
}
function show(){
var c = nav.className;
// do stuff
}
function reset(){
// do stuff
}
你可以將代碼封裝到一個(gè)對(duì)象中來(lái)避免上面的那種全局式編碼,這樣就可以將函數(shù)變成對(duì)象中的方法,將全局變量變成對(duì)象中的屬性。 你需要使用“名字+冒號(hào)”的方式來(lái)定義方法和屬性,并且需要在每個(gè)屬性或方法后面加上逗號(hào)作為分割符。
var myScript = {
nav:document.getElementById('nav'),
init:function(){
// do stuff
},
show:function(){
// do stuff
},
reset:function(){
// do stuff
}
}
相關(guān)文章
echarts實(shí)現(xiàn)晶體球面投影的實(shí)例教程
這篇文章主要給大家介紹了關(guān)于echarts實(shí)現(xiàn)晶體球面投影的相關(guān)資料,文中介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10javascript中節(jié)點(diǎn)的最近的相關(guān)節(jié)點(diǎn)訪問(wèn)方法
parentNode——父節(jié)點(diǎn);firstChild——第一個(gè)子節(jié)點(diǎn);lastChild——最后一個(gè)子節(jié)點(diǎn);previousSibling——緊挨著的前面的兄弟節(jié)點(diǎn);這樣就可以作短途旅行,訪問(wèn)當(dāng)前節(jié)點(diǎn)的某些相關(guān)節(jié)點(diǎn),感興趣的你可以參考下哈2013-03-03淺析JavaScript定時(shí)器setTimeout的時(shí)延問(wèn)題
這篇文章主要為大家詳細(xì)介紹了JavaScript中定時(shí)器setTimeout有最小時(shí)延的相關(guān)知識(shí),文中的示例代碼簡(jiǎn)潔易懂,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-11-11javascript數(shù)組常見(jiàn)操作方法實(shí)例總結(jié)【連接、添加、刪除、去重、排序等】
這篇文章主要介紹了javascript數(shù)組常見(jiàn)操作方法,結(jié)合實(shí)例形式總結(jié)分析了javascript數(shù)組的連接、添加、刪除、去重、排序等操作,代碼注釋附帶較為詳細(xì)的說(shuō)明,需要的朋友可以參考下2019-06-06原生JS實(shí)現(xiàn)圖片懶加載之頁(yè)面性能優(yōu)化
在項(xiàng)目開(kāi)發(fā)中,我們往往會(huì)遇到一個(gè)頁(yè)面需要加載很多圖片的情況。這篇文章主要介紹了頁(yè)面性能優(yōu)化原生JS實(shí)現(xiàn)圖片懶加載 ,需要的朋友可以參考下2019-04-04基于Javascript實(shí)現(xiàn)文件實(shí)時(shí)加載進(jìn)度的方法
不知道大家有沒(méi)有發(fā)現(xiàn)在現(xiàn)在的移動(dòng)頁(yè)面上,有很多情況需要加載大量的資源。但是移動(dòng)端的訪問(wèn)速度和pc還是有很大的差距,有些時(shí)候需要一些取巧的方式來(lái)提升用戶體驗(yàn),而實(shí)時(shí)顯示加載進(jìn)度就是其中一種。這篇文章就給大家分享了Javascript實(shí)現(xiàn)文件實(shí)時(shí)加載進(jìn)度的方法。2016-10-10JavaScript實(shí)現(xiàn)隨機(jī)數(shù)生成器(去重)
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)隨機(jī)數(shù)生成器,生成不重復(fù)的隨機(jī)數(shù),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10js使用split函數(shù)按照多個(gè)字符對(duì)字符串進(jìn)行分割的方法
這篇文章主要介紹了js使用split函數(shù)按照多個(gè)字符對(duì)字符串進(jìn)行分割的方法,實(shí)例分析了split函數(shù)的使用技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-03-03