最近做了個(gè)項(xiàng)目,其中有項(xiàng)目需求是要實(shí)現(xiàn)跑馬燈抽獎(jiǎng)效果,實(shí)現(xiàn)此功能主要用到j(luò)s相關(guān)知識(shí),廢話(huà)不多說(shuō),感興趣的朋友可以閱讀下全文。
開(kāi)始之前先來(lái)看上篇文章遺漏的兩個(gè)問(wèn)題和幾個(gè)知識(shí)點(diǎn),是自己重構(gòu)的過(guò)程中需要用到的:
1.移動(dòng)端1px像素線(xiàn)的問(wèn)題
對(duì)于設(shè)計(jì)師給我的手機(jī)端網(wǎng)頁(yè)的設(shè)計(jì)稿都是2倍圖。按照道理來(lái)說(shuō),在寫(xiě)網(wǎng)頁(yè)的時(shí)候,所有對(duì)象的實(shí)際尺寸都是會(huì)除2。但是對(duì)于1像素的線(xiàn)呢?
先來(lái)看兩張圖,設(shè)計(jì)稿的效果:
在三星 S4下的實(shí)際顯示效果:
可以看到這個(gè)時(shí)候1px的線(xiàn)竟然顯示不出來(lái)了。這個(gè)問(wèn)題是跟 S4手機(jī)的屏幕像素密度有關(guān)。關(guān)于屏幕像素密度和1px 線(xiàn)的關(guān)系有很多文章介紹,可以自行搜索了解。我這里的解決方案是,對(duì)1px 的線(xiàn)不做處理。是多少就寫(xiě)多少。就算我的基礎(chǔ)單位是rem,也不是其它單位。
{position: absolute;width: 13rem;height: 9.2rem;border:1px solid #000;}2.pc 端瀏覽器和移動(dòng)端瀏覽器容錯(cuò)率的差異
先來(lái)看一段代碼:
$('[node-type=row-a').find('div');很明顯可以發(fā)現(xiàn),我使用的選擇器是有語(yǔ)法錯(cuò)誤的。但是在瀏覽器中運(yùn)行會(huì)有什么結(jié)果呢?看下圖:

很明顯可以看出對(duì)于屬性選擇器,就算我有語(yǔ)法錯(cuò)誤,PC 端瀏覽器也是可以正確解析的。但是在手機(jī)端,這種寫(xiě)法是不能夠正確解析,代碼不能夠運(yùn)行。
所以寫(xiě)代碼的時(shí)候一定要注意一些小細(xì)節(jié)哈。。。
3.jQuery中選擇器的使用
在使用 jQuery 或者是 Zepto 的過(guò)程中最經(jīng)常使用的選擇器的寫(xiě)法就是下面這樣吧,
$('div.testClass')只是在$() 中寫(xiě)上自己需要的 Dom 節(jié)點(diǎn)的 class或者 ID 或 者使用屬性選擇器。
在查看 jQuery的文檔,對(duì)于$()會(huì)有這樣的描述:
jQuery([selector,[context]])
最重要的是看看對(duì) context (它也是我們平時(shí)使用中最容易忽略,但是卻非常有用的一個(gè)參數(shù))的描述:
默認(rèn)情況下, 如果沒(méi)有指定context參數(shù),$()將在當(dāng)前的 HTML document中查找 DOM 元素;如果指定了 context 參數(shù),如一個(gè) DOM 元素集或 jQuery 對(duì)象,那就會(huì)在這個(gè) context 中查找。在jQuery 1.3.2以后,其返回的元素順序等同于在context中出現(xiàn)的先后順序。
剛開(kāi)始學(xué)習(xí) JavaScript 那會(huì)兒,就聽(tīng)說(shuō)了操作 DOM 是很損耗瀏覽器性能,遍歷 DOM 也是很影響程序性能的。
如果我們?cè)谥付ǖ姆秶鷥?nèi)查找需要的 Dom 會(huì)不會(huì)比從整個(gè)document 中查找快很多。而且在我們寫(xiě) web 組件的過(guò)程中,一個(gè)頁(yè)面上組件可能出現(xiàn)很多次,那我們?cè)趺磁袛辔覀円僮髂膫€(gè)組件呢?這個(gè)context參數(shù)就會(huì)起到?jīng)Q定行的作用。具體請(qǐng)繼續(xù)看哇。。。
4.jQuery對(duì)象到數(shù)組的轉(zhuǎn)換
剛開(kāi)始學(xué)習(xí) jQuery的時(shí)候在一本書(shū)上看到一句話(huà):
jQuery對(duì)象就是一個(gè) JavaScript 數(shù)組。
而且在使用 jQuery的過(guò)程中,都會(huì)遇到,js對(duì)象轉(zhuǎn) jQuery對(duì)象,jQuery對(duì)象轉(zhuǎn) js對(duì)象。關(guān)于這些基礎(chǔ)不做過(guò)多介紹。
但是有時(shí)候我們會(huì)想在 jQuery對(duì)象上運(yùn)用一些原生Array對(duì)象的方法或者屬性。來(lái)看一個(gè)簡(jiǎn)單的例子:

由圖中的代碼運(yùn)行結(jié)果,可以知道在 jQuery對(duì)象上是沒(méi)有我們要使用reverse方法的。盡管test是一個(gè)數(shù)組。
那么我們?cè)趺崔k才可以讓 jQuery對(duì)象使用原生的 Array對(duì)象的方法呢?
4.1使用原型鏈擴(kuò)展
比如下面的代碼:
jQuery.prototype.reverse=function(){//一些操作}使用prototype來(lái)擴(kuò)展方法的時(shí)候,大家一直比較認(rèn)為是缺點(diǎn)的就是可能會(huì)污染已經(jīng)存在的原型鏈上的方法。還有就是訪(fǎng)問(wèn)方法的時(shí)候需要查找原型鏈。
4.2將 jQuery對(duì)象中的對(duì)象添加到數(shù)組中
看下面的代碼
var test = $('div.test');var a=[];$(test).each(function(){a.push($(this));});a.reverse();這樣就可以將 jQuery對(duì)象翻轉(zhuǎn)。
4.3使用 Array對(duì)象的 from()方法
這種方法也是自己在編寫(xiě)插件過(guò)程中使用的方法。看一下文檔描述:
Array.from() 方法可以將一個(gè)類(lèi)數(shù)組對(duì)象或可迭代對(duì)象轉(zhuǎn)換成真實(shí)的數(shù)組。
個(gè)人感覺(jué)使用這個(gè)代碼比較簡(jiǎn)潔。暫時(shí)還不知道有沒(méi)有性能的影響。繼續(xù)看下面的代碼:
var test = $('div.test');var a= Array.from(test);a.reverse();5.setInterval()和setTimeout()對(duì)程序性能的影響
因?yàn)閟etTimeout()和setInterval()這兩個(gè)函數(shù)在 JavaScript 中的實(shí)現(xiàn)機(jī)制完全一樣,這里只拿 setTimeout()驗(yàn)證
那么來(lái)看兩段代碼
var a ={test:function(){setTimeout(this.bbb,1000);},bbb:function(){console.log('----');}};a.test()輸出結(jié)果如下:

看下面的代碼輸出是什么
var a ={test:function(){setTimeout(function(){console.log(this);this.bbb();},1000);},bbb:function(){console.log('----');}};a.test();運(yùn)行這段代碼的時(shí)候,代碼報(bào)錯(cuò)

由以上的結(jié)果可以知道,當(dāng)我們?cè)谑褂胹etInterval()和setTimeout()的時(shí)候,在回掉中使用this的時(shí)候,this的作用域已經(jīng)發(fā)生了改變,并且指向了 window。
setTimeout(fn,0)的含義是,指定某個(gè)任務(wù)在主線(xiàn)程最早可得的空閑時(shí)間執(zhí)行,也就是說(shuō),盡可能早得執(zhí)行。它在”任務(wù)隊(duì)列”的尾部添加一個(gè)事件,因此要等到同步任務(wù)和”任務(wù)隊(duì)列”現(xiàn)有的事件都處理完,才會(huì)得到執(zhí)行。
意思就是說(shuō)在我們?cè)O(shè)置 setTimeout()之后,也可能不是立即等待多少秒之后就立即執(zhí)行回掉,而是會(huì)等待主線(xiàn)程的任務(wù)都處理完后再執(zhí)行,所以存在 “等待”超過(guò)自己設(shè)置時(shí)間的現(xiàn)象。同時(shí)也會(huì)存在異步隊(duì)列中已經(jīng)存在了其它的 setTimeout() 也是會(huì)等待之前的都執(zhí)行完再執(zhí)行當(dāng)前的。
看一個(gè) Demo:
setTimeout(function bbb(){},4000);function aaa(){setTimeout(function ccc(){},1000);}aaa();如果運(yùn)行上面的代碼,當(dāng)執(zhí)行完 aaa() 等待一秒后并不會(huì)立即執(zhí)行 ccc(),而是會(huì)等待 bbb() 執(zhí)行完再執(zhí)行 ccc() 這個(gè)時(shí)候離主線(xiàn)程運(yùn)行結(jié)束已經(jīng)4s 過(guò)去了。
以上內(nèi)容是針對(duì)JavaScript實(shí)現(xiàn)跑馬燈抽獎(jiǎng)活動(dòng)實(shí)例代碼解析與優(yōu)化(一),下篇繼續(xù)給大家分享JavaScript實(shí)現(xiàn)跑馬燈抽獎(jiǎng)活動(dòng)實(shí)例代碼解析與優(yōu)化(二),感興趣的朋友敬請(qǐng)關(guān)注。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注