主頁 >  其他 > 干貨收藏 | 如何優化前端性能?

干貨收藏 | 如何優化前端性能?

2020-10-19 00:19:57 其他

簡介:隨著前端的范疇逐漸擴大,深度逐漸下沉,富前端必然帶來的一個問題就是性能,特別是在大型復雜專案中,重前端業務可能因為一個小小的資料依賴,導致整個頁面卡頓甚至崩潰,本文基于Quick BI(資料可視化分析平臺)歷年架構變遷中性能的排查、解決和總結出的“個性”問題,嘗試總結整個前端層面相對“共性”的問題,提供一些前端性能解決思路,

image.png

一 引發性能問題原因?

引發性能問題的原因通常不是單方面緣由,特別是大型系統迭代多年后,長期積勞成疾造成,所以我們要必要分析找到癥結所在,并按瓶頸優先級逐個擊破,拿我們專案為例,大概分幾個方面:

1 資源包過大

通過Chrome DevTools的Network標簽,我們可以拿到頁面實際拉取的資源大小(如下圖):

image.png

經過前端高速發展,近幾年專案更新迭代,前端構建產物也在急劇增大,因為要業務先行,很多同學引入庫和編碼程序并沒有考慮性能問題,導致構建的包增至幾十MB,這樣帶來兩個顯著的問題:

  • 弱(普通)網路下,首屏資源下載耗時長
  • 資源解壓決議執行慢

對于第一個問題,基本上會影響所有移動端用戶,并且會耗費大量不必要的用戶帶寬,對客戶是一個經濟上的隱式損失和體驗損失,

對于第二個問題,會影響所有用戶,用戶可能因為等待時間過長而放棄使用,

下圖展示了延遲與用戶反應:

image.png

2 代碼耗時長

在代碼執行層面,專案迭代中引發的性能問題普遍是因為開發人員編碼質量導致,大概以下幾個緣由:

不必要的資料流監聽

此場景在hooks+redux的場景下會更容易出現,如下代碼:

const FooComponent = () => {
  const data = useSelector(state => state.fullData);
  return <Bar baz={data.bar.baz} />;
};

假設fullData是頻繁變更的大物件,雖然FooComponent僅依賴其.bar.baz屬性,fullData每次變更也會導致Foo重新渲染,

雙刃劍cloneDeep

相信很多同學在專案中都有cloneDeep的經歷,或多或少,特別是迭代多年的專案,其中難免有mutable型資料處理邏輯或業務層面依賴,需要用到cloneDeep,但此方法本身存在很大性能陷阱,如下:

// a.tsx
export const a = {
    name: 'a',
};
// b.tsx
import { a } = b;
saveData(_.cloneDeep(a)); // 假設需要克隆后落庫到后端資料庫

上方代碼正常迭代中是沒有問題的,但假設哪天 a 需要擴展一個屬性,保存一個ReactNode的參考,那么執行到b.tsx時,瀏覽器可能直接崩潰!

Hooks之Memo

hooks的發布,給react開發帶來了更高的自由度,同時也帶來了容易忽略的質量問題,由于不再有類中明碼標價的生命周期概念,組件狀態需要開發人員自由控制,所以開發程序中務必懂得react對hooks組件的渲染機制,如下代碼可優化的地方:

const Foo = () => { // 1. Foo可用React.memo,避免無props變更時渲染
    const result = calc(); // 2. 組件內不可使用直接執行的邏輯,需要用useEffect等封裝
    return <Bar result={result} />; // 3.render處可用React.useMemo,僅對必要的資料依賴作渲染
};

Immutable Deep Set

在使用資料流的程序中,很大程度我們會依賴lodash/fp的函式來實作immutable變更,但fp.defaultsDeep系列函式有個弊端,其實作邏輯相當于對原物件作深度克隆后執行fp.set,可能帶來一些性能問題,并且導致原物件所有層級屬性都被變更,如下:

const a = { b: { c: { d: 123 }, c2: { d2: 321 } } };
const merged = fp.defaultsDeep({ b: { c3: 3 } }, a);
console.log(merged.b.c === a.b.c); // 列印 false

3 排查路徑

對于這些問題來源,通過Chrome DevTools的Performance火焰圖,我們可以很清晰地了解整個頁面加載和渲染流程各個環節的耗時和卡頓點(如下圖):

image.png

當我們鎖定一個耗時較長的環節,就可以再通過矩陣樹圖往下深入(下圖),找到具體耗時較長的函式,

image.png

誠然,通常我們不會直接找到某個單點函式占用耗時非常長,而基本是每個N毫秒函式疊加執行成百上千次導致卡頓,所以這塊結合react除錯插件的Profile可以很好地幫助定位渲染問題所在:

image.png

如圖react組件被渲染的次數以及其渲染時長一目了然,

二 如何解決性能問題?

1 資源包分析

作為一名有性能sense的開發者,有必要對自己構建的產物內容保持敏感,這里我們使用到webpack提供的stats來作產物分析,

首先執行 webpack --profile --json > ./build/stats.json 得到 webpack的包依賴分析資料,接著使用 webpack-bundle-analyzer ./build/stats.json 即可在瀏覽器看到一張構建大圖(不同專案產物不同,下圖僅作舉例):

image.png

當然,還有一種直觀的方式,可以采用Chrome的Coverage功能來輔助判定哪些代碼被使用(如下圖):

image.png

最佳構建方式

通常來講,我們組織構建包的基本思路是:

  • 按entry入口構建,
  • 一個或多個共享包供多entry使用,

而基于復雜業務場景的思路是:

  • entry入口輕量化,
  • 共享代碼以chunk方式自動生成,并建立依賴關系,
  • 大資源包動態匯入(異步import),

webpack 4中提供了新的插件 splitChunks 來解決代碼分離優化的問題,它的默認配置如下:

module.exports = {
    //...
    optimization: {
        splitChunks: {
            chunks: 'async',
            minSize: 20000,
            minRemainingSize: 0,
            maxSize: 0,
            minChunks: 1,
            maxAsyncRequests: 30,
            maxInitialRequests: 30,
            automaticNameDelimiter: '~',
            enforceSizeThreshold: 50000,
            cacheGroups: {
                defaultVendors: {
                    test: /[\\/]node_modules[\\/]/,
                    priority: -10
                },
                default: {
                    minChunks: 2,
                    priority: -20,
                    reuseExistingChunk: true
                }
            }
        }
    }
};

根據上述配置,其分離chunk的依據有以下幾點:

  • 模塊被共享或模塊來自于node_modules,
  • chunk必須大于20kb,
  • 同一時間并行加載的chunk或初始包不得超過30,

理論上webpack默認的代碼分離配置已經是最佳方式,但如果專案復雜或耦合程度較深,仍然需要我們根據實際構建產物大圖情況,調整我們的chunk split配置,

解決TreeShaking失效

“你專案中有60%以上的代碼并沒有被使用到!”

treeshaking的初衷便是解決上面一句話中的問題,將未使用的代碼移除,

webpack默認生產模式下會開啟treeshaking,通過上述的構建配置,理論上應該達到一種效果“沒有被使用到的代碼不應該被打入包中”,而現實是“你認為沒有被使用的代碼,全部被打入Initial包中”,這個問題通常會在復雜專案中出現,其緣由就是代碼副作用(code effects),由于webpack無法判定某些代碼是否“需要產生副作用”,所以會將此類代碼打入包中(如下圖):

image.png

所以,你需要明確知道你的代碼是否有副作用,通過這句話判定:“關于‘副作用’的定義是,在匯入時會執行特殊行為的代碼(修改全域物件、立即執行的代碼等),而不是僅僅暴露一個 export 或多個 export,舉例說明,例如 polyfill,它影響全域作用域,并且通常不提供 export,”

對此,解決方法就是告訴webpack我的代碼沒有副作用,沒有被引入的情況下可以直接移除,告知的方式即:

在package.json中標記sideEffects為false,

或 在webpack配置中 module.rules 添加sideEffects過濾,

模塊規范

由此,要使得構建產物達到最佳效果,我們在編碼程序中約定了以下幾點模塊規范:

  • [必須] 模塊務必es6 module化(即export 和 import),
  • [必須] 三方包或資料檔案(如地圖資料、demo資料)超過 400KB 必須動態按需加載(異步import),
  • [禁止] 禁止使用export * as方式輸出(可能導致tree-shaking失效并且難以追溯),
  • [推薦] 盡可能引入包中具體檔案,避免直接引入整個包(如:import { Toolbar } from '@alife/foo/bar'),
  • [必須] 依賴的三方包必須在package.json中標記為sideEffects: false(或在webpack配置中標記),

2 Mutable資料

基本上通過Performance和React插件提供的除錯能力,我們基本可以定位問題所在,但對于mutable型的資料變更,我這里也結合實踐給出一些非標準除錯方式:

凍結定位法

眾所周知,資料流思想的產生緣由之一就是避免mutable資料無法追溯的問題(因為你無法知道是哪段代碼改了資料),而很多專案中避免不了mutable資料更改,此方法就是為了解決一個棘手的mutable資料變更問題而想出的方法,這里我暫時命名為“凍結定位法”,因為原理就是使用凍結方式定位mutable變更問題,使用相當tricky:

constob j= {
    prop: 42
};

Object.freeze(obj);

obj.prop=33; // Throws an error in strict mode

Mutable追溯

此方法也是為了解決mutable變更引發資料不確定性變更問題,用于實作排查的幾個目的:

  • 屬性在什么地方被讀取,
  • 屬性在什么地方被變更,
  • 屬性對應的訪問鏈路是什么,

如下示例,對于一個物件的深度變更或訪問,使用 watchObject 之后,不管在哪里設定其屬性的任何層級,都可以輸出變更相關的資訊(stack內容、變更內容等):

const a = { b: { c: { d: 123 } } };
watchObject(a);
const c =a.b.c;
c.d =0; // Print: Modify: "a.b.c.d"

watchObject 的原理即對一個物件進行深度 Proxy 封裝,從而攔截get/set權限,詳細可參考:
https://gist.github.com/wilsoncook/68d0b540a0fea24495d83fc284da9f4b

避免Mutable

通常像react這種技術堆疊,都會配套使用相應的資料流方案,其與mutable是天然對立的,所以在編碼程序中應該盡可能避免mutable資料,或者將兩者從設計上分離(不同store),否則出現不可預料問題且難以除錯

3 計算&渲染

最小化資料依賴

在專案組件爆炸式增長的情況下,資料流store內容層級也逐漸變深,很多組件依賴某個屬性觸發渲染,這個依賴項需要盡可能在設計時遵循最小化原則,避免像上方所述,依賴一個大的屬性導致頻繁渲染,

合理利用快取

(1)計算結果

在一些必要的cpu密集型計算邏輯中,務必采用 WeakMap 等快取機制,存盤當前計算終態結果或中間狀態,

(2)組件狀態

對于像hooks型組件,有必要遵循以下兩個原則:

  • 盡可能memo耗時邏輯,
  • 無多余memo依賴項,

避免cpu密集型函式

某些工具類函式,其復雜度跟隨入參的量級上升,而另外一些本身就會耗費大量cpu時間,針對這型別的工具,要盡量避免使用,若無法避免,也可通過 “控制入參內容(白名單)” 及 “異步執行緒(webworker等)”方式做到嚴控,

比如針對 _.cloneDeep ,若無法避免,則要控制其入參屬性中不得有參考之類的大型資料,

另外像最上面描述的immutable資料深度merge的問題,也應該盡可能控制入參,或者也可參考使用自研的immutable實作:
https://gist.github.com/wilsoncook/fcc830e5fa87afbf876696bf7a7f6bb1

const a = { b: { c: { d: 123 }, c2: { d2: 321 } } };
const merged = immutableDefaultsDeep(a, { b: { c3: 3 } });
console.log(merged === a); // 列印 false
console.log(merged.b.c === a.b.c); // 列印 true

三 寫在最后

以上,總結了Quick BI性能優化程序中的部分心得和經驗,性能是每個開發者不可繞過的話題,我們的每段代碼,都對標著產品的健康度,

原文鏈接:https://developer.aliyun.com/article/775774?

著作權宣告:本文內容由阿里云實名注冊用戶自發貢獻,著作權歸原作者所有,阿里云開發者社區不擁有其著作權,亦不承擔相應法律責任,具體規則請查看《阿里云開發者社區用戶服務協議》和《阿里云開發者社區知識產權保護指引》,如果您發現本社區中有涉嫌抄襲的內容,填寫侵權投訴表單進行舉報,一經查實,本社區將立刻洗掉涉嫌侵權內容,

轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/179133.html

標籤:其他

上一篇:SpringMVC概念和第一個程式

下一篇:谷歌十年掃地僧帶你學“三高!”:高并發+高性能+高可用

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 網閘典型架構簡述

    網閘架構一般分為兩種:三主機的三系統架構網閘和雙主機的2+1架構網閘。 三主機架構分別為內端機、外端機和仲裁機。三機無論從軟體和硬體上均各自獨立。首先從硬體上來看,三機都用各自獨立的主板、記憶體及存盤設備。從軟體上來看,三機有各自獨立的作業系統。這樣能達到完全的三機獨立。對于“2+1”系統,“2”分為 ......

    uj5u.com 2020-09-10 02:00:44 more
  • 如何從xshell上傳檔案到centos linux虛擬機里

    如何從xshell上傳檔案到centos linux虛擬機里及:虛擬機CentOs下執行 yum -y install lrzsz命令,出現錯誤:鏡像無法找到軟體包 前言 一、安裝lrzsz步驟 二、上傳檔案 三、遇到的問題及解決方案 總結 前言 提示:其實很簡單,往虛擬機上安裝一個上傳檔案的工具 ......

    uj5u.com 2020-09-10 02:00:47 more
  • 一、SQLMAP入門

    一、SQLMAP入門 1、判斷是否存在注入 sqlmap.py -u 網址/id=1 id=1不可缺少。當注入點后面的引數大于兩個時。需要加雙引號, sqlmap.py -u "網址/id=1&uid=1" 2、判斷文本中的請求是否存在注入 從文本中加載http請求,SQLMAP可以從一個文本檔案中 ......

    uj5u.com 2020-09-10 02:00:50 more
  • Metasploit 簡單使用教程

    metasploit 簡單使用教程 浩先生, 2020-08-28 16:18:25 分類專欄: kail 網路安全 linux 文章標簽: linux資訊安全 編輯 著作權 metasploit 使用教程 前言 一、Metasploit是什么? 二、準備作業 三、具體步驟 前言 Msfconsole ......

    uj5u.com 2020-09-10 02:00:53 more
  • 游戲逆向之驅動層與用戶層通訊

    驅動層代碼: #pragma once #include <ntifs.h> #define add_code CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) /* 更多游戲逆向視頻www.yxfzedu.com ......

    uj5u.com 2020-09-10 02:00:56 more
  • 北斗電力時鐘(北斗授時服務器)讓網路資料更精準

    北斗電力時鐘(北斗授時服務器)讓網路資料更精準 北斗電力時鐘(北斗授時服務器)讓網路資料更精準 京準電子科技官微——ahjzsz 近幾年,資訊技術的得了快速發展,互聯網在逐漸普及,其在人們生活和生產中都得到了廣泛應用,并且取得了不錯的應用效果。計算機網路資訊在電力系統中的應用,一方面使電力系統的運行 ......

    uj5u.com 2020-09-10 02:01:03 more
  • 【CTF】CTFHub 技能樹 彩蛋 writeup

    ?碎碎念 CTFHub:https://www.ctfhub.com/ 筆者入門CTF時時剛開始刷的是bugku的舊平臺,后來才有了CTFHub。 感覺不論是網頁UI設計,還是題目質量,賽事跟蹤,工具軟體都做得很不錯。 而且因為獨到的金幣制度的確讓人有一種想去刷題賺金幣的感覺。 個人還是非常喜歡這個 ......

    uj5u.com 2020-09-10 02:04:05 more
  • 02windows基礎操作

    我學到了一下幾點 Windows系統目錄結構與滲透的作用 常見Windows的服務詳解 Windows埠詳解 常用的Windows注冊表詳解 hacker DOS命令詳解(net user / type /md /rd/ dir /cd /net use copy、批處理 等) 利用dos命令制作 ......

    uj5u.com 2020-09-10 02:04:18 more
  • 03.Linux基礎操作

    我學到了以下幾點 01Linux系統介紹02系統安裝,密碼啊破解03Linux常用命令04LAMP 01LINUX windows: win03 8 12 16 19 配置不繁瑣 Linux:redhat,centos(紅帽社區版),Ubuntu server,suse unix:金融機構,證券,銀 ......

    uj5u.com 2020-09-10 02:04:30 more
  • 05HTML

    01HTML介紹 02頭部標簽講解03基礎標簽講解04表單標簽講解 HTML前段語言 js1.了解代碼2.根據代碼 懂得挖掘漏洞 (POST注入/XSS漏洞上傳)3.黑帽seo 白帽seo 客戶網站被黑帽植入劫持代碼如何處理4.熟悉html表單 <html><head><title>TDK標題,描述 ......

    uj5u.com 2020-09-10 02:04:36 more
最新发布
  • 2023年最新微信小程式抓包教程

    01 開門見山 隔一個月發一篇文章,不過分。 首先回顧一下《微信系結手機號資料庫被脫庫事件》,我也是第一時間得知了這個訊息,然后跟蹤了整件事情的經過。下面是這起事件的相關截圖以及近日流出的一萬條資料樣本: 個人認為這件事也沒什么,還不如關注一下之前45億快遞資料查詢渠道疑似在近日復活的訊息。 訊息是 ......

    uj5u.com 2023-04-20 08:48:24 more
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:47:46 more
  • vulnhub_Earth

    前言 靶機地址->>>vulnhub_Earth 攻擊機ip:192.168.20.121 靶機ip:192.168.20.122 參考文章 https://www.cnblogs.com/Jing-X/archive/2022/04/03/16097695.html https://www.cnb ......

    uj5u.com 2023-04-20 07:46:20 more
  • 從4k到42k,軟體測驗工程師的漲薪史,給我看哭了

    清明節一過,盲猜大家已經無心上班,在數著日子準備過五一,但一想到銀行卡里的余額……瞬間心情就不美麗了。最近,2023年高校畢業生就業調查顯示,本科畢業月平均起薪為5825元。調查一出,便有很多同學表示自己又被平均了。看著這一資料,不免讓人想到前不久中國青年報的一項調查:近六成大學生認為畢業10年內會 ......

    uj5u.com 2023-04-20 07:44:00 more
  • 最新版本 Stable Diffusion 開源 AI 繪畫工具之中文自動提詞篇

    🎈 標簽生成器 由于輸入正向提示詞 prompt 和反向提示詞 negative prompt 都是使用英文,所以對學習母語的我們非常不友好 使用網址:https://tinygeeker.github.io/p/ai-prompt-generator 這個網址是為了讓大家在使用 AI 繪畫的時候 ......

    uj5u.com 2023-04-20 07:43:36 more
  • 漫談前端自動化測驗演進之路及測驗工具分析

    隨著前端技術的不斷發展和應用程式的日益復雜,前端自動化測驗也在不斷演進。隨著 Web 應用程式變得越來越復雜,自動化測驗的需求也越來越高。如今,自動化測驗已經成為 Web 應用程式開發程序中不可或缺的一部分,它們可以幫助開發人員更快地發現和修復錯誤,提高應用程式的性能和可靠性。 ......

    uj5u.com 2023-04-20 07:43:16 more
  • CANN開發實踐:4個DVPP記憶體問題的典型案例解讀

    摘要:由于DVPP媒體資料處理功能對存放輸入、輸出資料的記憶體有更高的要求(例如,記憶體首地址128位元組對齊),因此需呼叫專用的記憶體申請介面,那么本期就分享幾個關于DVPP記憶體問題的典型案例,并給出原因分析及解決方法。 本文分享自華為云社區《FAQ_DVPP記憶體問題案例》,作者:昇騰CANN。 DVPP ......

    uj5u.com 2023-04-20 07:43:03 more
  • msf學習

    msf學習 以kali自帶的msf為例 一、msf核心模塊與功能 msf模塊都放在/usr/share/metasploit-framework/modules目錄下 1、auxiliary 輔助模塊,輔助滲透(埠掃描、登錄密碼爆破、漏洞驗證等) 2、encoders 編碼器模塊,主要包含各種編碼 ......

    uj5u.com 2023-04-20 07:42:59 more
  • Halcon軟體安裝與界面簡介

    1. 下載Halcon17版本到到本地 2. 雙擊安裝包后 3. 步驟如下 1.2 Halcon軟體安裝 界面分為四大塊 1. Halcon的五個助手 1) 影像采集助手:與相機連接,設定相機引數,采集影像 2) 標定助手:九點標定或是其它的標定,生成標定檔案及內參外參,可以將像素單位轉換為長度單位 ......

    uj5u.com 2023-04-20 07:42:17 more
  • 在MacOS下使用Unity3D開發游戲

    第一次發博客,先發一下我的游戲開發環境吧。 去年2月份買了一臺MacBookPro2021 M1pro(以下簡稱mbp),這一年來一直在用mbp開發游戲。我大致分享一下我的開發工具以及使用體驗。 1、Unity 官網鏈接: https://unity.cn/releases 我一般使用的Apple ......

    uj5u.com 2023-04-20 07:40:19 more