我优化了进度条,页面性能竟提高了70%,那些BAT大厂的前端面试官到底在想些什么

来简单演示一下这个进度条的样子

0opakkask9kklana669029.gif

为什么说这种写法不太好呢?因为我们是通过定时器来快速递增变量progress以此来实现进度增加的,变量每次改变都会驱动视图重新计算渲染,这必然是性能很差的(说实话,我在体验这个demo的时候,肉眼可见的小卡顿)

除此之外呢?其实还有一个造成卡顿的原因,你们不妨猜猜看,我们放到最后一起讲,想知道答案的小伙伴可以直接滑到下面

推荐的写法

==============================================================

这里推荐的就是我在阅读代码时看到的比较优秀的方案了,接下来分享给大家

组件部分

// index.jsx

import { useState } from ‘react’

import ‘./index.css’

let totalTime = 3000 // 假设视频播放为3s

function App() {

const [isPlay, setIsPlay] = useState(false) // 是否播放

const [count, setCount] = useState(0) // 播放次数

const [type, setType] = useState(0) // 使用哪个动画。0: @keyframes play; 1: @keyframes replay;

// 暂停 && 播放

const handleVideo = () => setIsPlay(!isPlay);

// 重播

const replay = () => {

setIsPlay(true)

setType(type ? 0 : 1)

}

// 动画结束时触发的事件

const end = () => {

setCount(count + 1) // 播放次数 +1

replay() // 重新开始播放

}

return (

{ isPlay ? ‘暂停’ : ‘播放’ }

重播

{ 播放次数为:${count} }

className={progress ${isPlay ? 'play' : 'pause'}}

style={{

animationDuration: ${totalTime}ms,

animationName: ${type ? 'replay' : 'play'}

}}

onAnimationEnd={end} // 动画结束时的事件

/>

)

}

样式部分

@keyframes play {

to {

width: 100%;

}

}

@keyframes replay {

to {

width: 100%;

}

}

.container {

height: 10px;

border-radius: 5px;

border: 1px solid black;

}

.progress {

height: 100%;

width: 0;

background-color: red;

animation-timing-function: linear;

}

.progress.play { /* 使animation动画启动 */

animation-play-state: running;

}

.progress.pause { /* 使animation动画暂停 */

animation-play-state: paused;

}

我们设置了两个@keyframes动画是为了在使进度条重新播放时可以做一个切换,即点击 “重播” 时,直接切换到另一个动画,就可以实现进度条从0开始递增

同时我们还设置了两个类名的样式,分别用于控制动画的播放和暂停

播放完成时,播放次数+1的功能可以通过事件animationend来监听即可

同样的,来看一下这套方案的效果图(跟前一套方案功能一模一样)

0opakkask9kklana669029.gif

对比一下前一套方案,你就能知道这种写法不需要去一直修改数据来驱动视图的改变,减少了框架内的大量计算,提升了不少的性能

缺陷

===========================================================

第二种方案虽然性能很好,但是与第一种方案一样,存在另外一个隐藏的性能问题,这也是我在排查前同事代码性能问题时所发现的。

缺陷:这两种方案都会引发频繁的重排和重绘

可以借助chrome devtools performance来验证一下页面的情况

11010asdakkkakasd69029.gif

小小的一个进度条触发了那么那么多次重排和重绘,那么它到底有什么影响呢?来简单回顾一下重排和重绘的影响

重排:浏览器需要重新计算元素的几何属性,而且其他元素的几何属性或位置可能也会因此改变受到影响。

重绘:不是所有的DOM变化都影响元素的几何属性,如果改变元素的背景色并不影响它的宽度和高度,这种情况,只会发生一次重绘,而不会发生重排,因为元素的布局没改变

所以知道了重排和重绘造成的严重问题后,我们马上对其进行分析优化

极致的优化

==============================================================

先来看看一个非常常见的图

image.png

页面的渲染,大体上走的就是这5个流程。当然也有办法跳过中间某些步骤,例如避免LayoutPaint

再来回顾一下有哪些方法会引起重排和重绘吧

触发重排的因素:添加或删除可见的DOM元素、改变元素位置、元素的尺寸改变(包括:外边距、内边距、边框、高度等)、内容改变(如:文本改变或图片被另外一个不同尺寸的图片替代)、浏览器窗口尺寸的改变、通过display: none隐藏⼀个DOM节点等

触发重绘的因素:重排必定触发重绘(重要)、通过visibility: hidden隐藏⼀个DOM节点、修改元素背景色、修改字体颜色等

那么我们前面写的代码中到底是哪里触发了重排和重绘呢?简单检查一下,不难发现两种方案都是在不停改变元素的width,元素的宽度一改变必然会引起重排和重绘,更何况是超频繁的改变呢!

解决方案:启用GPU加速,避开重排和重绘的环节,将进度条单独提升到一个图层,即不影响其它元素

就单独针对第二种方案进行优化吧~我们只需要改动其css内容即可(标注出即为改动处)

@keyframes play { /* 通过transform来启用GPU加速,跳过重排重绘阶段 */

0% {

transform: translateX(-50%) scaleX(0); /* 用 scaleX 来代替 width */

}

to {

transform: translateX(0) scaleX(1);

}

}

@keyframes replay {

0% {

transform: translateX(-50%) scaleX(0);

}

to {

transform: translateX(0) scaleX(1);

}

}

.container {

height: 10px;

border-radius: 5px;

border: 1px solid black;

}

.progress {

height: 100%;

width: 100%; /* 初始宽度为100%,因为我们要对其缩放 */

background-color: red;

will-change: transform; /* 通过will-change告知浏览器提前做好优化准备 */

animation-timing-function: linear;

}

.progress.play {

animation-play-state: running;

}

.progress.pause {

animation-play-state: paused;

}

这里简单解释一下translateXscaleX的数值设置。设置进度条width: 100%,我们通过scaleX(0.5)将其缩放一半,可以发现进度条长度为容器的一半且居中,此时我们就需要通过translateX(-25%)将其向左平移到最左端,为什么是-25%呢?因为进度条占了容器的一半且居中,表明左右的留白正好分别是(100% - 50%) / 2 = 25%,所以也不难得知当初始状态scaleX(0)时,translateX的值为-(100% - 0%) / 2 = -50%

这么做了以后,我们再次用performance检验一下

kasjdaskdj0022asd69029.gif

可以很明显地看到页面重排重绘的次数减少了很多很多,剩余的基本都是页面最基本的重排和重绘了。

有人要说我标题党了,接下来给你们展示一下到底优化了多少性能

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024c (备注前端)
img

文末

逆水行舟不进则退,所以大家要有危机意识。

同样是干到35岁,普通人写业务代码划水,榜样们深度学习拓宽视野晋升管理。

这也是为什么大家都说35岁是程序员的门槛,很多人迈不过去,其实各行各业都是这样都会有个坎,公司永远都缺的高级人才,只用这样才能在大风大浪过后,依然闪耀不被公司淘汰不被社会淘汰。

为了帮助大家更好温习重点知识、更高效的准备面试,特别整理了《前端工程师核心知识笔记》电子稿文件。

内容包括html,css,JavaScript,ES6,计算机网络,浏览器,工程化,模块化,Node.js,框架,数据结构,性能优化,项目等等。

269页《前端大厂面试宝典》

包含了腾讯、字节跳动、小米、阿里、滴滴、美团、58、拼多多、360、新浪、搜狐等一线互联网公司面试被问到的题目,涵盖了初中级前端技术点。

CodeChina开源项目:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

前端面试题汇总

序员的门槛,很多人迈不过去,其实各行各业都是这样都会有个坎,公司永远都缺的高级人才,只用这样才能在大风大浪过后,依然闪耀不被公司淘汰不被社会淘汰。

为了帮助大家更好温习重点知识、更高效的准备面试,特别整理了《前端工程师核心知识笔记》电子稿文件。

内容包括html,css,JavaScript,ES6,计算机网络,浏览器,工程化,模块化,Node.js,框架,数据结构,性能优化,项目等等。

269页《前端大厂面试宝典》

包含了腾讯、字节跳动、小米、阿里、滴滴、美团、58、拼多多、360、新浪、搜狐等一线互联网公司面试被问到的题目,涵盖了初中级前端技术点。

CodeChina开源项目:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

前端面试题汇总

  • 9
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值