NVIDIA 初创加速计划,免费加速您的创业启动 了解详情
写点什么

浅析 Web 前端性能优化

  • 2020-02-14
  • 本文字数:2699 字

    阅读完需:约 9 分钟

浅析Web前端性能优化

什么是 WEB 前端呢?就是用户电脑的浏览器,所做的一切事情。我们来看看用户访问网站,浏览器都做了哪些事情。


输入网址 –> 解析域名 -> 请求页面 -> 解析页面并发送页面中的资源请求 -> 渲染资源 -> 输出页面 -> 监听用户操作 -> 重新渲染。


通过上面可以看出浏览器分为三部分来实现用户的使用,我们就从这三个部分来浅析如何提升 WEB 前端性能。


一、 请求


浏览器为了减少请求传输,实现了自己的缓存机制。浏览器缓存就是把一个已经请求过的 Web 资源拷贝一份副本存储在浏览器中。当再次请求相同的 URL 时,先去查看缓存,如果有本地缓存,浏览器缓存机制会根据验证机制(Etag)和过期机制(Last-Modified)进行判断是使用缓存,还是从服务器传输资源文件。具体流程如下图所示:


1506321631051056382.png


浏览器的请求有些是并发的,有些是阻塞的。图片、CSS、接口的请求是并发,JS 文件是阻塞的。请求 JS 的时候,浏览器会中断渲染进程,等待 JS 文件加载解析完毕,再重新渲染。所以要把 JS 文件放在页面的最后。


JS 也可以通过两种方式由阻塞改成并行。一种是通过创建 script 标签,插入 DOM 中;另一种是在 Script 标签中增加 async 属性。


每种浏览器对同一域名并发的数量有限制,IE6/7 是 2,IE9 是 10,其他常见的浏览器是 6,所以减少资源请求数量和使用多域名配置资源文件,能大大提高网站性能。


减少资源请求数量的方法,大致有以下几种:


1、 通过打包工具,合并资源,减少资源数量。就是开发版本是很多个资源文件,部署的时候,按类合并成几个文件来输出。在实现模块管理的同时,实现统一输出。


2、 CSS 中,使用 css sprite 减少图片请求数量。


3、 通过延迟加载技术,在用户无感知的情况下请求资源。


4、 通过服务器配置,实现一次请求,返回多个资源文件,如淘宝 CDN 那样。


除了减少请求数量,也可以使用 CDN 镜像,来减少网络节点,实现快速响应。使用了 CDN 的请求,会根据用户所处的地理位置,找寻最近的 CDN 节点,如果请求是新的,则从资源服务器拷贝到节点,然后再返回给客户端。如果请求已经存在,则直接从节点返回客户端。


通过上面我们了解的缓存机制,如果我们部署上线的时候,是需要刷新缓存的。普通缓存通过强刷就能改过来,而 CDN 缓存则需要通过改变 URL 来实现。同时我们不可能要求用户按着 Ctrl 来刷新,所以通过打包工具,在部署的时候,统一更改 URL 是最有效的方式。而不常变更的库文件,比如 echart、jquery,则不建议更改。


二、 传输


从服务器往客户端传输,可以开启 gzip 压缩来提高传输效


Gzip 有从 1-10 的十个等级。越高压缩的越小,但压缩使用的服务器硬件资源就越多。根据实践,等级为 5 的时候最均衡,此时压缩效果是 100k 可以压缩成 20k。


三、 渲染


浏览器在加载了 html 后,就会一边解析,一边根据解析出来


结果进行资源请求,并生成 DOM 树。而加载完毕的 CSS,则被渲染引擎根据生成好的 DOM 树,来生成渲染树。等所有资源解析完毕计算好 layout 后,向浏览器界面绘制。随着用户操作,JS 会修改 DOM 节点或样式,重新绘制和重新排列。重新绘制指的是绘制 DOM 节点对应的渲染节点,重新排列是指重新计算这些节点在浏览器界面的位置。很显然,重排是非常耗性能的。我们要做的是减少重排的次数。


生成 DOM 树的时候,我们可以通过减少 DOM 节点来优化性能。最初都是用 table 布局,节点深度和数量相当复杂,性能很差。同样 CSS 作为层叠样式表,层级也不可太深,不然遍历的成本很高。另外 CSS 的 expression 属性相当耗性能,能不用则不用。动画效果能用 CSS 写的就不用 JS 写,渲染引擎不一样,性能损耗也不一样。


上面说的是解析渲染的过程,我们再接着说说用户交互操作的过程。用户操作就会导致重绘和重排,重排一定会引起重绘,而重绘不一定会引起重排。到底怎样会引起重排呢?简单的定义,DOM 结构的变化,以及 DOM 样式中几何属性的变化,就会导致重排。几何属性顾名思义,就是宽、高、边框、外补丁、内补丁等俗称盒模型的属性。同时还有 offset 之类的边距属性。


重排是最耗能的,减少重排的方法有:


1、 如果需要多次改变 DOM,则先在内存中改变,最后一次性的插入到 DOM 中。


2、 同上一条,如果多次改变样式,合成一条,再插入 DOM 中。


3、 由于 position 的值为 absoute 和 fixed 时候,是脱离文档流的,操作此类 DOM 节点,不会引起整页重排。所以动画元素设置 position 使其脱离文档流。


4、 当 DOM 节点的 display 等于 none 的时候,是不会存在于渲染树的,所以如果有比较复杂的操作,先使其 display 等于 none,等待所有操作完毕后,再将 display 设成 block,这样就只重排两次。


5、 获取会导致重排的属性值时,存入变量,再次使用时就不会再次重排。获取这些属性会导致重排:offsetTop、offsetLeft、offsetWidth、offsetHeight、scrollTop、scrollLeft、scrollWidth、scrollHeight、clientTop、clientLeft、clientWidth、clientHeight


以上就是浏览器如何把资源变成肉眼所见的页面的,除了上述根据浏览器流程而总结出来的性能优化,我们还需要看看 javascript 作为程序,需要的优化。先来看看 javascript 的垃圾回收机制。


Javascript 的引擎会在固定的时间间隔,将不再使用的局部变量注销掉,释放其所占的内存。而闭包的存在,将使引用一直存在,无法被释放掉。全局变量的生命周期直至浏览器卸载页面才会结束。所以一般来讲,内存溢出就是由于全局变量的不释放和闭包引起。为了防止内存溢出,我们可以做的方法有:


1、 业务代码放在匿名立即执行函数里面,执行完毕会立即释放掉。


2、 少用全局变量,同时用完的变量手动注销掉。


3、 使用回调来代替闭包访问内部属性


4、 当不可避免使用闭包时,慎重的对待其中的细节。不用的时候注销掉。


5、 通过浏览器自带的工具 profiles,来检查内存活动情况。如果是波浪型的,说明正常。如果是倾斜式渐进上涨的,说明有内存不会被释放,需要检查相应的函数。


最后再说一点,函数里返回异步取的值,经常有人这么:


Var getList = function(){ $.ajax().then(function(data){
Return data;
}) };
Var users = getList();
复制代码


毫无疑问,由于函数内的返回是异步的,所以返回只能是 undefined,而不是想要的 data。于是为了实现返回 data,就把 ajax 的 async 属性设置成了 false,由异步改为同步,来获取到 data。然而最大的问题来了,同步是会中断渲染进程的,也就是请求返回的等待中,整个页面是卡死的,用户操作也不会有响应。这个问题真正的解决方案是返回 promise 对象,而不是把异步改成同步。


本文转载自宜信技术学院网站。


原文链接:http://college.creditease.cn/detail/143


2020-02-14 18:55822

评论

发布
暂无评论
发现更多内容

基于 Web SDK 实现视频通话场景 | 声网 SDK 教程

声网

视频 SDK 教程

李宏毅《机器学习》丨7. Conclusion(总结)

AXYZdong

7月月更

图的存储结构与方法(二)

乔乔

7月月更

做一个有职业操守的软件匠人

Bruce Talk

技术 敏捷 TDD Agile

7大专题详解SpringBoot,阿里这套SpringBoot全栈笔记真香

Java永远的神

Java 程序员 面试 程序人生 springboot

全新出品!Github总榜排行第七的SpringCloud生态全栈笔记我粉了

Java全栈架构师

Java 程序员 面试 微服务 SpringCloud

@Entity 里面的 JPA 注解

Damon

7月月更

springMvc参数获取

沃德

Java 7月月更

界面设计四大原则

空城机

设计模式 7月月更

mysql进阶(十九)SQL语句如何精准查找某一时间段的数据

No Silver Bullet

MySQL 7月月更 精确查找

OpenIM重大优化-消息按需加载 一致性缓存 uniapp发布

Geek_1ef48b

自动驾驶产品化竞备开启:百度Apollo如何定义量产车?

脑极体

MySQL数据库索引

技术小生

索引 7月月更

节流和防抖的说明和实现

南极一块修炼千年的大冰块

7月月更

深入浅出边缘云 | 1. 概述

俞凡

架构 边缘计算 网络 深入浅出边缘云

一时跳槽一时爽,一直跳槽一直爽?

KEY.L

7月月更

Vue Router 守卫

程序员海军

Vue 7月月更

类的基础

GalaxyCreater

python小知识-代码规范最佳实践

AIWeker

7月月更 pyhon小知识

Protocol buffers 的问题和滥用

HoneyMoose

面试官:MySQL 数据库查询慢,除了索引问题还可能是什么原因?

Java全栈架构师

Java MySQL 数据库 面试 后端

【Go实现】实践GoF的23种设计模式:观察者模式

元闰子

Go 设计模式 观察者模式 Go 语言

OSI模型第一层:物理层,基石般的存在!

wljslmz

物理层 网络技术 OSI模型 7月月更

多线程&高并发(全网最新:面试题+导图+笔记)面试手稳心不慌

冉然学Java

Java 编程 多线程并发 高并发系统 资料分享

12张图+6K字图解ZGC垃圾回收器及调优技巧

程序员小毕

程序员 面试 程序人生 ZGC JVM

参与开源社区还有证书拿?

玩转Devop和研发效能DevStream/DevLake

GitHub 开源 开发者 证书

Docker安装Elasticsearch、ik分词器、可视化工具

宁在春

Docker Elastic Stack 7月月更

IntelliJ IDEA使用

GalaxyCreater

Java IDEA

Java开发环境配置 / Vscode搭建

攻城狮杰森

Java jdk 7月月更

SpringBoot项目使用Tomcat为什么让大厂禁止?

Java程序员

Java 程序员 Sprint Boot

作为一名后台开发人员,你必须知道的两种过滤器

C++后台开发

后台开发 后端开发 Linux服务器开发 C/C++后台开发 C/C++开发

浅析Web前端性能优化_行业深度_马宗泽_InfoQ精选文章