Vue筆記六:Vue項目的性能優化之路

Vue筆記六:Vue項目的性能優化之路

我最近也經常面試外包同事。面試的時候,總會有個問題,“你說一下性能優化的手段”。百分之八十的人都會說,壓縮js和css之類的。顯然這些都是必須做的,而且已經根本不是主要的性能優化的關鍵點。如果你只會說這些,只能說明你是個過時的前端工程師。

性能優化過程中,我們需要面對的更多是DMS解析過程,服務器緩存和瀏覽器緩存機制。

gzip壓縮

在所有的web前端項目,靜態資源基本都放在cdn上,gzip的壓縮是非常必要的,它直接改變了js文件的大小,減少兩到三倍。

參考加速nginx: 開啟gzip和緩存,nginx的gzip配置非常簡單,在你對應的域名底下,添加下面的配置,重啟服務即可。gzip_comp_level的值大于2的時候并不明顯,建議設置在1或者2之間。

# 開啟gzip
gzip on;
# 啟用gzip壓縮的最小文件,小于設置值的文件將不會壓縮
gzip_min_length 1k;
# gzip 壓縮級別,1-10,數字越大壓縮的越好,也越占用CPU時間,后面會有詳細說明
gzip_comp_level 2;
# 進行壓縮的文件類型。javascript有多種形式。其中的值可以在 mime.types 文件中找到。
gzip_types text/plain application/javascript application/x-javascript text/css application/xml text/javascript application/x-httpd-php image/jpeg image/gif image/png;
# 是否在http header中添加Vary: Accept-Encoding,建議開啟
gzip_vary on;
# 禁用IE 6 gzip
gzip_disable "MSIE [1-6]\.";

服務器緩存

為了提高服務器獲取數據的速度,nginx緩存著靜態資源是非常必要的。如果是測試服務器對html應該不設置緩存,而js等靜態資源環境因為文件尾部會加上一個hash值,這可以有效實現緩存的控制。

location ~* ^.+\.(ico|gif|jpg|jpeg|png)$ { 
  access_log   off; 
  expires      30d;
}
location ~* ^.+\.(css|js|txt|xml|swf|wav)$ {
  access_log   off;
  expires      24h;
}
location ~* ^.+\.(html|htm)$ {
  expires      1h;
}

瀏覽器緩存

瀏覽器緩存是通過html的頭文件中的meta來控制。http-equiv是一個專門針對http的頭文件,可以向瀏覽器傳回一些有用的信息。與之對應的content,是各個參數的變量值。

HTTP 1.0

在HTTP1.0中通過Pragma控制頁面緩存,可以設置為Pragmano-cache。在不讓瀏覽器或中間緩存服務器緩存頁面的情況下,通常設置的值為no-cache,不過這個值不這么保險,通常還加上Expires置為0來達到目的。Expires可以用于設定網頁的到期時間。一旦網頁過期,必須到服務器上重新傳輸獲取新的頁面信息。PS:內容必須使用GMT的時間格式。

<meta http-equiv="Pragma" content="no-cache">
<meta http-equiv="Expires" content="0">

HTTP 1.1

在HTTP1.1中通過Cache-Control控制頁面緩存,可以設置為no-cacheprivateno-storemax-agemust-revalidate等,默認為private。

<meta http-equiv="Cache-Control" content="no-cache">
  • public 瀏覽器和緩存服務器都可以緩存頁面信息
  • private 對于單個用戶的整個或部分響應消息,不能被共享緩存處理。這允許服務器僅僅描述當用戶的部分響應消息,此響應消息對于其他用戶的請求無效
  • no-cache 瀏覽器和緩存服務器都不應該緩存頁面信息
  • no-store 請求和響應的信息都不應該被存儲在對方的磁盤系統中,不使用緩存
  • must-revalidate 對于客戶機的每次請求,代理服務器必須想服務器驗證緩存是否過時
  • max-age 客戶機可以接收生存期不大于指定時間(以秒為單位)的響應
  • min-fresh 客戶機可以接收響應時間小于當前時間加上指定時間的響應

Last-Modified和Etags

Last-Modified服務器端文件響應頭,描述最后修改時間。當瀏覽器再次進行請求時,會向服務器傳送If-Modified-Since報頭,詢問時間點之后資源是否被修改過,從而區分200和304的請求狀態碼,304則選擇瀏覽器緩存。

Etags不同的是,ETag是根據實體內容生成一段hash字符串,是標識資源的狀態。它由服務端產生來判斷文件是否有更新。

參考資料:

JS分包

前面說的兩部分都可以說是偏后端的活,如果真的從前端方面考慮,我們可能會分包入手。正因為vue的腳手架搭建的項目,webpack的配置當中就包含了壓縮js,css和html的壓縮。所以,當我們的單頁面越做越大的情況下,首要的一步就是分包。

vue官方稱gzip壓縮后只有20kb,但是你普通的打包方式也有100kb,再加上你自己的邏輯代碼,整體包的體積也挺大的。直接影響首屏頁面加載的效率。下面介紹一下兩種分包的方法:

  • external 把包排除,使用cdn資源
  • dll 打包

vue,vuex和vue-router

在webpack配置文件中external設置,把這三個場用包排除這個操作,主要是把這三個包從vendor.js分開。

最后當然需要在html標簽上添加上額外cdn的link或者script。

DLL打包

這種打包方式專門引用webpack官方的DllPluginDllReferencePlugin。DllPlugin會生成一個dll包的代碼指紋manifest,管理額外的打包。而在項目生成的過程中,DllReferencePlugin會參考manifest的內容去打包。額外生成的js文件應該被放置在vue項目的文件當中的static文件夾底下,以便于代碼部署。

參考PaicFE/vue-multi中的配置文件webpack.dll.config.js的寫法。

預加載

預加載技術(prefetch)是在用戶需要前我們就將所需的資源加載完畢,不是所有瀏覽器都支持,主要是Chrome瀏覽器。

DNS prefetch 分析這個頁面需要的資源所在的域名,瀏覽器空閑時提前將這些域名轉化為 IP 地址,真正請求資源時就避免了上述這個過程的時間。----HTML5 prefetch

由于域名轉換成為IP的過程是非常耗時的一個過程,DNS prefetch可以減少這部分的時間。

<meta http-equiv='x-dns-prefetch-control' content='on'>
<link rel='dns-prefetch' >
<link rel='dns-prefetch' >
<link rel='dns-prefetch' >
<link rel='dns-prefetch' >
<link rel='dns-prefetch' >

預加載也可以對某個靜態資源起到專門的作用。

<link rel='subresource' href='libs.js'>

預渲染(pre-rendering)是這個頁面會提前加載好用戶即將訪問的下一個頁面。

<link rel='prerender' >

vue組件keep-alive

如果你做用一個大型web的spa的時候,你有很多router,對應的是很多個頁面。在頁面的快速切換中,為了保證頁面加載的效率,除了緩存機制之外,vue的keep-alive組件可以幫的上忙。

它會把組件保存在瀏覽器內存當中,方便你快速切換。

百度的lavas項目中就在vue-router當中使用keep-alive的組件,用它包裹著router-view。使用了keep-alive的組件內的數據將會保留,“是否需要重新同步數據”可以在vue-router的鉤子中路由所帶的參數執行判斷。

Promise請求

es6的其中一個特性就是原生支持promise。在這里,我先不說異步編程里的generatoraync/await的屬性。它們功能的實現都是基于promise。

Promise的特點在于:

  • 減少回調函數
  • 串并行處理
  • 代碼的優雅

這里特別講一下,ES6在性能優化上可以使用promise或者async/await去壓縮請求時間。在過去,很多jquery的頁面在調用接口請求都是一個接口等另一個接口,串行執行所有請求,最后在完成最后的回調函數,如此類推。這樣的寫法會直接導致“回調地獄”。即使你用vue-resource,我也review到非常多的“回調地獄”的情況。為了從根本上解決這個問題并提高開發效率,我建議優先使用promise。(async/await不急著投入使用),考慮到還有很多同事還在高效地開發業務代碼。

現在的vue-resource已經支持promise的寫法,為了更好地讓技術向后發展,我建議將pagekit/vue-resource替換稱為mzabriskie/axioswebmodules/jsonpaxios是可以同時滿足服務端和瀏覽器端,同構的寫法有助于以后將技術棧往SSR(服務端渲染)發展。jsonp這個庫則是為了兼容jsonp的請求需要,需要對它進行了promise的封裝。

export function getJsonp(urlHost, key, data, _params) {
  return new Promise((resolve, reject) => {
    let url = urlHost + key;
    if (data) url += `?${querystring.stringify({ ...data, temp: new Date().getTime() })}`;
    const params = _params || { timeout: 15000 };
    if (!params.timeout) params.timeout = 15000;
    jsonp(url, params, (err, res) => {
      if (err) {
        reject(err);
      } else {
        resolve(res);
      }
    });
  });
}

Promise的使用需要避免以下的寫法,

promise.then(function(value) {
  // success
}, function(error) {
  // failure
});

盡量使用鏈式寫法,

promise.then(function(value) {
  // step1
}).then(function(value){
  // step2
}).catch(function(value){
  // failure
})

并行的操作主要是Promise.all(),它可以將Promise操作的數組并行執行完成然后在進行串行的操作。Promise.race()則是返回并行請求中最先返回的請求的那個結果。它們的使用可以有效地壓縮數據獲取的時間。

擴展閱讀

轉載,請表明出處。總目錄前段收集器

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,333評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,491評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,263評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,946評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,708評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,409評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,939評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,774評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,641評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,872評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,650評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內容

  • 網站優化離不開前后端的互相協作,但是對于前端工程師來說,在保證后端技術方案不變時,能不能只利用前端技術來優化網站呢...
    留七七閱讀 6,362評論 0 31
  • Yahoo!的Exceptional Performance團隊為改善Web性能帶來最佳實踐。他們為此進行了一系列...
    拉風的老衲閱讀 1,863評論 0 1
  • 問答題47 /72 常見瀏覽器兼容性問題與解決方案? 參考答案 (1)瀏覽器兼容問題一:不同瀏覽器的標簽默認的外補...
    _Yfling閱讀 13,776評論 1 92
  • 租了房子,丟了手機,欠債,生活開銷……一點點全是經濟壓力,至少此刻我是很有壓力的。想想晚上多出來的時間都是用錢買來...
    慕容珊閱讀 409評論 0 0
  • 作者名片:老貓枕咸魚 全文目錄:《未懸年》簡介及目錄 預判,或者稱之為預斷,對被判斷者來說是并不十分公平的推斷,這...
    老貓枕咸魚閱讀 530評論 10 15