阿里妹导读
写在前面
性能优化对于提供卓越的用户体验至关重要,钉钉终端团队特别关注用户体验。我们团队采用了一系列创新的性能优化措施,显著提升了首次有意义绘制(FMP)和首次内容绘制(FCP)的性能指标。其中,利用快照方案,结合用户的本地存储能力,我们能够进一步提高页面性能。快照方案是在完成常规手段前端优化(如优化首屏加载体积、实施懒加载、渲染优化和缓存提升等)和资源离线处理之后的又一重要步骤,旨在更迅速地向用户展示页面内容。
钉钉的 PC 工作台通过应用快照技术加速了页面渲染,并从此经验中提炼出了一个通用的快照 SDK,使得其他页面也能轻松集成此技术,从而提高其性能。不仅限于钉钉端内应用本身,同样适用于解决端外等各种场景下的性能提升需求。
接下来,我们将探讨快照技术如何增强页面性能和用户体验,如何在业务中集成快照方案,以及我们的通用快照解决方案的技术细节。
快照方案概述
快照 从概念上,这个词从摄影领域借鉴而来,在计算机领域是指在某个特定时间点对系统、数据或配置状态的完整副本,而系统或程序可以利用这一份记录实现快速恢复、启动优化等。
基于快照的性能体验优化手段,主要利用了存储在 客户端 本地的 快照 资源,加速页面速度,提升用户体验。
快照方案对前端性能提升的作用:改善首屏显示速度,减少白屏时间。
快照优点:能很好的保存每个用户 千人千面 的信息,并且有可能和 SSR+流式渲染结合。
使用案例和效果演示
钉钉标准 PC 工作台首页
数据效果
钉钉机器人管理
数据效果
快照 SDK
简介
接入指南
安装
tnpm install @ali/snapshot-dd-webpack-plugin --save-dev
# 或
ayarn add @ali/snapshot-dd-webpack-plugin --dev
使用
快速体验快照能力版
const SnapshotDDWebpackPlugin = require('@ali/snapshot-dd-webpack-plugin');
module.exports = {
// ...
plugins: [
new HtmlWebpackPlugin(),
// 新增代码
new SnapshotDDWebpackPlugin(),
]
// ...
};
检测快照是否开启成功
-
修改webpack配置后, tnpm start 重启项目。 -
查看element元素中是否有id为html-snapshot的快照节点,检查其中内容是否符合预期(快照一般会在第二次打开页面才展示快照,第一次打开页面会存储快照)。
精细化调整配置版
const SnapshotDDWebpackPlugin = require('@ali/snapshot-dd-webpack-plugin');
module.exports = {
// ...
plugins: [
new HtmlWebpackPlugin(),
// 新增代码
new SnapshotDDWebpackPlugin({
// 可选配置选项config,详细可配置项说明见下文IConfig
// 页面根元素id(即react全局挂载容器id),默认取dingapp
// rootId: 'mytestid',
// 默认为false,使用indexDB存储方式,核心业务可配置true使用localStorage
// useLocalStorage: true,
// 灰度配置, 仅支持钉钉端内
// grayConfig: {
// disable: '你的general模块key', // 禁用快照开关key
// mobile: 'win_snapshot_enable',
// pc: 'pc_snapshot_enable',
// mac: '你的general模块key', // 控制mac端能力灰度,仅在mac端生效
// win: 'win_snapshot_enable', // 控制win端能力灰度,仅在win端生效
// android: 'win_snapshot_enable', // 控制android端能力灰度,仅在android端生效
// ios: 'win_snapshot_enable', // 控制ios端能力灰度,仅在ios端生效
// },
// 自定义处理快照内容,将以该方法返回的内容作为页面快照内容
// handleSnapshotHtml: (data) => '
test',
// 快照内容替换,可配置对快照做微调处理:挖空、替换可能发生改变产生闪烁的元素
// snapshotSlotContentMap: {
// '.dtm-button-secondary': ``,
// },
// 保存快照成功回调,可进行埋点等操作
// takeSnapShotCallback: (data) => console.log('takeSnapShotCallback data:', data),
// 快照时机,默认onload之后100ms
// takeSnapShotDelay: 1000,
// 配置检测到该元素上屏时,执行隐藏快照逻辑,例如'.your-class #yourId'
// hideSnapshotSelector: '.dtm-button-secondary',
// 移除快照成功回调,可进行埋点等操作
// hideSnapShotCallback: (data) => console.log('hideSnapShotCallback data:', data),
// 未配置hideSnapshotSelector时,会自动检测 FCP后2s 隐藏快照,配置隐藏的delay时间
// 未配置hideSnapshotSelector时,会自动检测 FCP后2s 隐藏快照,配置隐藏的delay时间
// hideSnapShotFCPDelay: 2000,
// 配置不支持自动FCP的delay隐藏时间,默认3s
// hideSnapShotNotSupportFCPDelay: 2000,
// debug模式配置,debug模式会有更多log打点
// debug: true,
})
]
// ...
};
可配置项IConfig
interface IConfig {
// 页面根元素id(即react全局挂载容器id),默认取dingapp,若非dingapp,请指定
rootId?: string;
// 默认为false,使用indexDB存储方式,核心业务可配置true使用localStorage
useLocalStorage?: boolean;
// 灰度配置, 仅支持钉钉端内
grayConfig?: {
disable?: string; // 禁用快照
pc?: string; // 控制PC端能力灰度,仅在PC端生效
mobile?: string; // 控制移动端能力灰度,仅在移动端生效
android?: string; // 控制android端能力灰度,仅在android端生效
ios?: string; // 控制ios端能力灰度,仅在ios端生效
mac?: string; // 控制mac端能力灰度,仅在mac端生效
win?: string; // 控制win端能力灰度,仅在win端生效
}
// 自定义处理快照内容,将以该方法返回的内容作为页面快照内容
handleSnapshotHtml?: string; // (html: string) => string;
// 快照内容替换,可配置对快照做微调处理:挖空、替换可能发生改变产生闪烁的元素
snapshotSlotContentMap?: {
[querySelector: string]: string; // key为任意selector,value为HTML内容的字符串表示
};
// 保存快照成功回调,可进行埋点等操作
takeSnapShotCallback?: string; // (html?: string) => void;
// 快照时机,默认onload之后100ms
takeSnapShotDelay?: number;
// 配置检测到该元素上屏时,执行隐藏快照逻辑,例如'.your-class #yourId'
hideSnapshotSelector?: string;
// 移除快照成功回调,可进行埋点等操作
hideSnapShotCallback?: string; // () => void;
// 未配置hideSnapshotSelector时,会自动检测 FCP后2s 隐藏快照,配置隐藏的delay时间
hideSnapShotFCPDelay?: number;
// 配置不支持自动FCP的delay隐藏时间,默认3s
hideSnapShotNotSupportFCPDelay?: number;
// debug模式配置,debug模式会有更多log打点
debug?: boolean;
}
灰度开关配置
grayConfig?: {
disable?: string; // 禁用快照
pc?: string; // 控制PC端能力灰度,仅在PC端生效
mobile?: string; // 控制移动端能力灰度,仅在移动端生效
android?: string; // 控制android端能力灰度,仅在android端生效
ios?: string; // 控制ios端能力灰度,仅在ios端生效
mac?: string; // 控制mac端能力灰度,仅在mac端生效
win?: string; // 控制win端能力灰度,仅在win端生效
}
自定义用法
注意事项
-
请确保您的webpack配置文件中,HtmlWebpackPlugin已经配置好,否则快照功能无法生效; -
请确保您的项目中,页面根元素id若非dingapp,请在配置中指定您的rootId,否则快照功能无法生效; -
请确保您的项目中,将css以内联 形式打包到html中已经配置好,否则快照功能中样式可能错乱; -
默认配置中保存快照、展示快照、移除快照时机均为默认值,若需更加精细化效果呈现,请在配置中调整;
实现方案
快照的作用是什么?
1 工作原理
-
生成快照:把页面中关键元素的数据缓存到本地存储 -
展示快照:页面加载过程中,从本地存储中提取出之前保存的快照,并将其作为临时的DOM覆盖在实际页面之上 -
移除快照:当页面的真实DOM渲染完毕,移除上层的DOM快照,让用户得以看到最新渲染的页面内容
Step1 生成快照
什么时候生成快照?
关键元素
数据
快照形式内容对比
-
磁盘
存储位置对比
Step2 展示快照
什么时候展示快照?
Step3 移除快照
什么时候隐藏快照?
快照 SDK 效果
<html>
<head>
<script>
// 增加参数传递部分
window.__DD_SNAPSHOT_CONFIG__ = {
"rootId": "Root",
"debug": true,
"useLocalStorage": true
}
script>
script>
head>
<body class="bg-common" data-spm="28107366">
<div id="html-snapshot" style="position: absolute; left: 0; right: 0; top: 0; z-index: 9999999;">div>
<script>
// 快照展示逻辑
!async function() {
const n = function() {
const n = new URL(window.location.href)
, {pathname: t, search: e, hash: o} = n;
let a = e;
e.startsWith("?") && (a = e.substring(1));
const r = new URLSearchParams(a)
, s = [];
for (const [n,t] of r)
!["dd_mini_app_id", "pc_slide", "dd_darkmode", "dd_progress", "dtaction"].includes(n) && s.push(t);
const i = [t.replace(/\.html$/, ""), s.join("_"), o].filter((n=>n)).join("-").replace(/[^a-zA-Z0-9-]/g, "_");
return window.__ddSnapshotKey = `ddSnapshotKey_${i}`,
window.__ddSnapshotKey
}()
, t = await async function(n) {
let t;
return t = window.__DD_SNAPSHOT_CONFIG__ && window.__DD_SNAPSHOT_CONFIG__.useLocalStorage ? localStorage.getItem(n) : await async function(n) {
try {
const t = await function(n, t, e) {
return new Promise(((n,o)=>{
const a = indexedDB.open("SnapshotDatabase8");
a.onerror = n=>{
o("数据库打开失败")
}
,
a.onsuccess = a=>{
const r = a.target.result.transaction(t, "readonly").objectStore(t).get(e);
r.onerror = n=>{
o("获取数据失败")
}
,
r.onsuccess = t=>{
n(t.target.result)
}
}
}
))
}(0, "snapshot", n);
return t
} catch (n) {
console.error("[snapshot]: Error while loading snapshot:", n)
}
}(n),
t || ""
}(n)
, e = document