前端性能優(yōu)化-代碼優(yōu)化

一、JS開(kāi)銷和如何縮短解析時(shí)間

JS開(kāi)銷相比其他資源開(kāi)銷大的很多,js開(kāi)銷有加載跋涣,加載完成后需要編譯解析缨睡、最后是執(zhí)行。



如何縮短解析時(shí)間陈辱?
1.code splitting代碼拆分奖年,按需加載
2.tree shaking代碼減重(webpack自動(dòng)進(jìn)行)
3.避免超過(guò)1kb的行間腳本
4.避免長(zhǎng)任務(wù)
5.使用rAF和rIC進(jìn)行時(shí)間調(diào)度

二、配合V8 有效優(yōu)化代碼

V8編譯原理



1沛贪、分詞/詞法分析:
這個(gè)過(guò)程會(huì)將字符串分割為有意義的代碼塊陋守,這些代碼塊稱之為詞法單元。例如變量的聲明:
var a = 2;
這行代碼會(huì)被分為以下詞法單元:var利赋、a水评、=、2(空格算不算詞法單元取決于空格對(duì)于該編程語(yǔ)言是否具有意義)媚送;這些零散的詞法單元會(huì)組成一個(gè)詞法單元流(數(shù)組)進(jìn)行解析中燥。

2、解析/與法分析:
這個(gè)過(guò)程會(huì)將詞法單元流轉(zhuǎn)換成一棵抽象語(yǔ)法樹(shù)(Abstract Syntax Tree塘偎,AST)在線解析工具疗涉。
"var a = 2;"的詞法單元流就會(huì)被解析為下面的AST:
var a = 2;對(duì)應(yīng)的AST



3、代碼生成:
將AST轉(zhuǎn)化為可執(zhí)行的代碼吟秩。

V8的優(yōu)化機(jī)制:
1咱扣、腳本流
2、字節(jié)碼緩存
3涵防、懶解析

三闹伪、函數(shù)優(yōu)化.

V8這樣的JS引擎,默認(rèn)對(duì)函數(shù)進(jìn)行懶解析(lazy parsing):只有當(dāng)這個(gè)函數(shù)真正被調(diào)用的時(shí)候壮池,才會(huì)去解析函數(shù)體偏瓤。這樣的解析方式會(huì)對(duì)性能有很好的提升。但是實(shí)際開(kāi)發(fā)中有的情況需要JS立即去執(zhí)行火窒,此時(shí)如果是懶解析的函數(shù)但是發(fā)現(xiàn)需要立即執(zhí)行硼补,于是又快速的進(jìn)行了一個(gè)饑餓解析(eager parsing),這樣的話會(huì)先進(jìn)行了一次饑餓解析熏矿,然后又進(jìn)行了一次懶解析已骇,執(zhí)行效率反而降低了。

那么如何告訴解析器某個(gè)函數(shù)需要立即解析:
只需加一對(duì)括號(hào):

const add = ((a,b) => a+b)    // 這樣的作用是告訴解析器票编,當(dāng)看到這個(gè)函數(shù)的時(shí)候就對(duì)這個(gè)函數(shù)進(jìn)行解析
const n1 = 1
const n2 = 2
add(n1,n2)

這樣當(dāng)后面需要調(diào)用的時(shí)候就可以直接調(diào)用了
有個(gè)問(wèn)題:
當(dāng)把代碼進(jìn)行壓縮時(shí)(uglyfy)褪储,可能會(huì)把這對(duì)括號(hào)去掉
解決:Optimize.js』塾颍可以在壓縮以后把括號(hào)加回來(lái)
uglyfy后來(lái)把這些問(wèn)題也解決了鲤竹,需要兼顧老版本的可以添加

避免反優(yōu)化

performance.mark('start') //性能測(cè)量標(biāo)記
add = (a,b) =>  a + b
 
const num1 = 0
const num2 = 2
 
for(let i = 0; i < 1000000; i++) {
  add(num1,num2)
}
 
add(num1,'1dsd')   // V8已經(jīng)做了對(duì)于add的優(yōu)化。如參數(shù)類型改變。所以要撤掉已做的優(yōu)化辛藻,會(huì)造成一些延遲碘橘。反優(yōu)化
 
for(let i = 0; i < 1000000; i++) {
  add(num1,num2)
}
performance.mark('end')

四、對(duì)象優(yōu)化

1吱肌,以相同順序初始化對(duì)象成員痘拆,避免隱藏類的調(diào)整

class RectArea { // HC0
    constructor(l, w) {
        this.l = l; // HC1
        this.w = w; // HC2
    }
}
 
const rect1 = new RectArea(3,4); // 創(chuàng)建了隱藏類HC0, HC1, HC2
const rect2 = new RectArea(5,6); // 相同的對(duì)象結(jié)構(gòu),可復(fù)用之前的所有隱藏類
 
const car1 = {color: 'red'}; // HC0
car1.seats = 4; // HC1
 
const car2 = {seats: 2}; // 沒(méi)有可復(fù)用的隱藏類氮墨,創(chuàng)建HC2
car2.color = 'blue'; // 沒(méi)有可復(fù)用的隱藏類纺蛆,創(chuàng)建HC3

2,實(shí)例化后规揪,避免添加新屬性

const car1 = {color: 'red'}; // In-object 屬性
car1.seats = 4; // Normal/Fast 屬性桥氏,存儲(chǔ)在property store里,需要通過(guò)描述數(shù)組間接查找

3猛铅,盡量使用Array 代替 array-like 對(duì)象字支。轉(zhuǎn)成數(shù)組再進(jìn)行操作。

Array.prototype.forEach.call(arrObj, (value, index) => { // 不如在真實(shí)數(shù)組上效率高
  console.log(`${ index }: ${ value }`);
});
 
const arr = Array.prototype.slice.call(arrObj, 0); // 轉(zhuǎn)換的代價(jià)比影響優(yōu)化小
arr.forEach((value, index) => {
  console.log(`${ index }: ${ value }`);
});

4奸忽,不要讀取超過(guò)數(shù)組的長(zhǎng)度

function foo(array) {
  for (let i = 0; i <= array.length; i++) { // 越界比較
    if(array[i] > 1000) { // 1.沿原型鏈的查找 2.造成undefined與數(shù)進(jìn)行比較
        console.log(array[i]); // 業(yè)務(wù)上無(wú)效祥款、出錯(cuò)
    }   
  }

5,避免元素類型的轉(zhuǎn)換月杉。eg:數(shù)組元素類型的統(tǒng)一

const array = [3, 2, 1]; // PACKED_SMI_ELEMENTS
array.push(4.4); // PACKED_DOUBLE_ELEMENTS

五、HTML優(yōu)化

1.減少iframes使用
2.壓縮空白符
3.避免節(jié)點(diǎn)深層級(jí)嵌套
4.避免table布局(開(kāi)銷大抠艾,維護(hù)麻煩)
5.刪除注釋
6.CSS&JS盡量外鏈(導(dǎo)致html過(guò)大苛萎,不好優(yōu)化)CSS放在頭部 JS放在body底部
7.刪除元素默認(rèn)屬性

六、css 優(yōu)化

1检号、選擇器

.highlight-list
 .list:nth-last-child(1) > #box a

我們都建議定義一個(gè)單一的樣式類來(lái)表示該元素腌歉,如上面的第一行
而不要使用 偽類 等各種選擇器 來(lái)確定元素
因?yàn)槲覀兌贾繡SS選擇器解析 從右到左進(jìn)行的,讀取到a的時(shí)候齐苛,會(huì)獲取所有的a翘盖,然后根據(jù)nth-last-child(1)>這個(gè)條件進(jìn)行過(guò)濾,得到滿足條件的a 凹蜂,然后再往左走根據(jù).list這個(gè)條件過(guò)濾馍驯,得到最終的結(jié)果a
但是最新的瀏覽器的研究表明,兩者性能相差并不大玛痊,可以說(shuō)是完全一樣汰瘫,不是影響性能的主要因素

2、降低CSS對(duì)渲染的阻塞
低css的大小
提前對(duì)css文件進(jìn)行下載
對(duì)首屏展示有關(guān)的css先加載擂煞,無(wú)關(guān)的進(jìn)行延遲加載

3混弥、利用GPU進(jìn)行完成動(dòng)畫(huà)
will-change: transform; transform等之前講過(guò)不會(huì)進(jìn)行布局和重繪,會(huì)單獨(dú)在一層对省,GPU直接進(jìn)行干預(yù)

4.使用contain屬性

.news li {
  padding: 10px;
  contain: layout;
}

為什么要使用這個(gè)contain屬性呢
就比如 一個(gè)ul 有多個(gè)li 實(shí)現(xiàn)一個(gè)新聞列表
當(dāng)我們對(duì)第一個(gè)li插入一個(gè)新東西蝗拿,瀏覽器并不知道這個(gè)操作會(huì)不會(huì)影響其他元素的布局晾捏,這時(shí)候?yàn)g覽器會(huì)對(duì)其他元素進(jìn)行重新的位置等方面的計(jì)算,這時(shí)候開(kāi)銷很大哀托。但是我們開(kāi)發(fā)者肯定這個(gè)操作不會(huì)對(duì)其他元素造成影響惦辛。
此時(shí)我們就可以利用contain:layout,與瀏覽器進(jìn)行溝通萤捆。告訴瀏覽器我這個(gè)盒子的布局與外部沒(méi)有任何關(guān)系裙品。里面怎么變化不會(huì)影響到外面,同時(shí)外面怎么變化不會(huì)影響到里面俗或。
這樣瀏覽器就清楚了市怎,會(huì)單獨(dú)處理操作的元素,而不會(huì)對(duì)其他元素進(jìn)行處理辛慰,減低了很大一部分開(kāi)銷区匠。

  1. font-display字體加載,顯示方式
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末帅腌,一起剝皮案震驚了整個(gè)濱河市驰弄,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌速客,老刑警劉巖戚篙,帶你破解...
    沈念sama閱讀 211,348評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異溺职,居然都是意外死亡岔擂,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,122評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門浪耘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)乱灵,“玉大人,你說(shuō)我怎么就攤上這事七冲⊥匆校” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,936評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵澜躺,是天一觀的道長(zhǎng)蝉稳。 經(jīng)常有香客問(wèn)我,道長(zhǎng)苗踪,這世上最難降的妖魔是什么颠区? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,427評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮通铲,結(jié)果婚禮上毕莱,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好朋截,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,467評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布蛹稍。 她就那樣靜靜地躺著,像睡著了一般部服。 火紅的嫁衣襯著肌膚如雪唆姐。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,785評(píng)論 1 290
  • 那天廓八,我揣著相機(jī)與錄音奉芦,去河邊找鬼。 笑死剧蹂,一個(gè)胖子當(dāng)著我的面吹牛声功,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播宠叼,決...
    沈念sama閱讀 38,931評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼先巴,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了冒冬?” 一聲冷哼從身側(cè)響起伸蚯,我...
    開(kāi)封第一講書(shū)人閱讀 37,696評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎简烤,沒(méi)想到半個(gè)月后剂邮,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,141評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡横侦,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,483評(píng)論 2 327
  • 正文 我和宋清朗相戀三年抗斤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片丈咐。...
    茶點(diǎn)故事閱讀 38,625評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖龙宏,靈堂內(nèi)的尸體忽然破棺而出棵逊,到底是詐尸還是另有隱情,我是刑警寧澤银酗,帶...
    沈念sama閱讀 34,291評(píng)論 4 329
  • 正文 年R本政府宣布辆影,位于F島的核電站,受9級(jí)特大地震影響黍特,放射性物質(zhì)發(fā)生泄漏蛙讥。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,892評(píng)論 3 312
  • 文/蒙蒙 一灭衷、第九天 我趴在偏房一處隱蔽的房頂上張望次慢。 院中可真熱鬧,春花似錦、人聲如沸迫像。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,741評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)闻妓。三九已至菌羽,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間由缆,已是汗流浹背注祖。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留均唉,地道東北人是晨。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像浸卦,于是被迫代替她去往敵國(guó)和親署鸡。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,492評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容