前端白屏问题_深入理解前端性能监控

在同样的网络环境下,有两个同样能满足你的需求的网站,一个唰的一下就加载出来了,另一个白屏转圈转了半天内容才出来,如果让你选择,你会用哪一个?

页面的性能问题是前端开发中一个重要环节,但一直以来我们没有比较好的手段,来检测页面的性能。直到W3C性能小组引入的新的API window.performance,目前IE9以上的浏览器都支持。

它是一个浏览器中用于记录页面加载和解析过程中关键时间点的对象。放置在global环境下,通过JavaScript可以访问到它。

使用性能API

你可以通过以下方法来探测和兼容performance:

var performance = window.performance || 
window.msPerformance ||
window.webkitPerformance;
if (performance) {
// 你的代码
}

先来了解一下performance的结构:

e1c604ab3e16e1db93624b9bfe54ee7b.png

performance.memory是显示此刻内存占用情况,它是一个动态值,其中:

usedJSHeapSize表示:JS 对象(包括V8引擎内部对象)占用的内存数

totalJSHeapSize表示:可使用的内存 

jsHeapSizeLimit表示:内存大小限制 通常,usedJSHeapSize不能大于totalJSHeapSize,如果大于,有可能出现了内存泄漏。

performance.navigation显示页面的来源信息,其中: redirectCount表示:如果有重定向的话,页面通过几次重定向跳转而来。

默认为0 type表示页面打开的方式, 0 表示 TYPE_NAVIGATENEXT 正常进入的页面(非刷新、非重定向等)。

1 表示 TYPE_RELOAD 通过             window.location.reload() 刷新的页面。

2 表示 TYPE_BACK_FORWARD 通过浏览器的前进后退按钮进入的页面(历史记录) 。

255 表示 TYPE_UNDEFINED 非以上方式进入的页面。

performance.onresourcetimingbufferfull 属性是一个在resourcetimingbufferfull事件触发时会被调用的 event handler 。它的值是一个手动设置的回调函数,这个回调函数会在浏览器的资源时间性能缓冲区满时执行。

performance.timeOrigin是一系列时间点的基准点,精确到万分之一毫秒。

performance.timing是一系列关键时间点,它包含了网络、解析等一系列的时间数据。

9ce2c6c15bc731754f810a0120f35712.png

下面是对这些时间点进行解释

timing: {
// 同一个浏览器上一个页面卸载(unload)结束时的时间戳。如果没有上一个页面,这个值会和fetchStart相同。
navigationStart: 1543806782096,
// 上一个页面unload事件抛出时的时间戳。如果没有上一个页面,这个值会返回0。
unloadEventStart: 1543806782523,

// 和 unloadEventStart 相对应,unload事件处理完成时的时间戳。如果没有上一个页面,这个值会返回0。
unloadEventEnd: 1543806782523,

// 第一个HTTP重定向开始时的时间戳。如果没有重定向,或者重定向中的一个不同源,这个值会返回0。
redirectStart: 0,

// 最后一个HTTP重定向完成时(也就是说是HTTP响应的最后一个比特直接被收到的时间)的时间戳。
// 如果没有重定向,或者重定向中的一个不同源,这个值会返回0.
redirectEnd: 0,

// 浏览器准备好使用HTTP请求来获取(fetch)文档的时间戳。这个时间点会在检查任何应用缓存之前。
fetchStart: 1543806782096,

// DNS 域名查询开始的UNIX时间戳。
//如果使用了持续连接(persistent connection),或者这个信息存储到了缓存或者本地资源上,这个值将和fetchStart一致。
domainLookupStart: 1543806782096,

还有一些其他的参数就不一一列举了,这些参数非常有用,可以帮助我们获取页面的Domready时间、onload时间、白屏时间等,以及单个页面资源在从发送请求到获取到rsponse各阶段的性能参数。

对我们比较有用的页面性能数据大概包括如下几个,这些参数是通过上面的                  performance.timing各个属性的差值组成的,它是精确到毫秒的一个值,计算方法如下:

  • 重定向耗时 : redirectEnd - redirectStart

  • DNS查询耗时 : domainLookupEnd - domainLookupStart

  • TCP链接耗时 : connectEnd - connectStart

  • HTTP请求耗时 : responseEnd - responseStart

  • 解析dom树耗时 : domComplete - domInteractive

  • 白屏时间 : responseStart - navigationStart

  • DOMready时间 :

     domContentLoadedEventEnd - navigationStart

  • onload时间:loadEventEnd - navigationStart,也即是onload回调函数执行的时间。

如何优化?

是隐式的 DNS Prefetch。如果想对页面中没有出现的域进行预获取,那么就要使用显示的 DNS Prefetch 了。下图是DNS Prefetch的方法:



腾讯网
"dns-prefetch" href="//mat1.gtimg.com" />
"dns-prefetch" href="//inews.gtimg.com" />
"dns-prefetch" href="//wx.qlogo.cn" />
"dns-prefetch" href="//coral.qq.com" />
"dns-prefetch" href="//pingjs.qq.com" />

**TCP请求优化**:TCP的优化大都在服务器端,前端能做的就是尽量减少TCP的请求数,也就是减少HTTP的请求数量。http 1.0 默认使用短连接,也是TCP的短连接,也就是客户端和服务端每进行一次http操作,就建立一次连接,任务结束就中断连接。

这个过程中有3次TCP请求握手和4次TCP请求释放。减少TCP请求的方式有两种,一种是资源合并,对于页面内的图片、css和js进行合并,减少请求量。

另一种使用长链接,使用http1.1,在HTTP的响应头会加上 Connection:keep-alive,当一个网页打开完成之后,连接不会马上关闭,再次访问这个服务时,会继续使用这个长连接。

这样就大大减少了TCP的握手次数和释放次数。或者使用Websocket进行通信,全程只需要建立一次TCP链接。

**HTTP请求优化**:使用内容分发网络(CDN)和减少请求。使用CDN可以减少网络的请求时延,CDN的域名不要和主站的域名一样,这样会防止访问CDN时还携带主站cookie的问题,对于网络请求,可以使用fetch发送无cookie的请求,减少http包的大小。也可以使用本地缓存策略,尽量减少对服务器数据的重复获取。

**渲染优化**:在浏览器端的渲染过程,如大型框架,vue和react,它的模板其实都是在浏览器端进行渲染的,不是直出的html,而是要走框架中相关的框架代码才能去渲染出页面,这个渲染过程对于首屏就有较大的损耗,白屏的时间会有所增加。在必要的情况下可以在服务端进行整个html的渲染,从而将整个html直出到我们的浏览器端,而非在浏览器端进行渲染。

03d80e6c8608832c6f3e122e6835671e.png

还有一个问题就是,在默认情况下,JavaScript 执行会“阻止解析器”,当浏览器遇到一个 script 外链标记时,DOM 构建将暂停,会将控制权移交给 JavaScript 运行时,等脚本下载执行完毕,然后再继续构建 DOM。而且内联脚本始终会阻止解析器,除非编写额外代码来推迟它们的执行。

我们可以把 script 外链加入到页面底部,也可以使用 defer 或 async 延迟执行。defer 和 async 的区别就是 defer 是有序的,代码的执行按在html中的先后顺序,而 async 是无序的,只要下载完毕就会立即执行。或者使用异步的编程方法,比如settimeout,也可以使用多线webworker,它们不会阻碍 DOM 的渲染。



资源性能API

performance.timing记录的是用于分析页面整体性能指标。如果要获取个别资源(例如JS、图片)的性能指标,就需要使用Resource Timing API。 **performance.getEntries()**方法,包含了所有静态资源的数组列表;每一项是一个请求的相关参数有name,type,时间等等。下图是chrome显示腾讯网的相关资源列表。

bfed56d0b81052af56f6c06b10cd224b.png

可以看到,与 performance.timing 对比: 没有与 DOM 相关的属性,新增了nameentryTypeinitiatorTypeduration四个属性。它们是

  • name表示: 资源名称,也是资源的绝对路径,可以通过           performance.getEntriesByName(name属性的值),来获取这个资源加载的具体属性。

  • entryType表示:资源类型 "resource",还有“navigation”, “mark”, 和 “measure”另外3种。

fa8abe75d72231b352c8e280e07ec48f.png

  • initiatorType表示:请求来源"link",即表示标签还有            script即 ,“img”即标签,“css”比如background的url方式加载资源以及“redirect”即重定向 等。

6f8cf8915a621acca0d06b88ea75297c.png

  • duration表示:加载时间,是一个毫秒数字。 受同源策略影响,跨域资源获取到的时间点,通常为0,如果需要更详细准确的时间点,可以单独请求资源通过performance.timing获得。或者资源服务器开启响应头Timing-Allow-Origin,添加指定来源站点,如下所示:

Timing-Allow-Origin: https://qq.com

方法集合

除了performance.getEntries之             外,performance还包含一系列有用的方法。如下图

9398e96a1668e0be71d4eeff395d7b33.png

performance.now()  返回一个当前页面执行的时间的时间戳,用来精确计算程序执行时间。与 Date.now() 不同的是,它使用了一个浮点数,返回了以毫秒为单位,小数点精确到微秒级别的时间,更加精准。并且不会受系统程序执行阻塞的影响,performance.now() 的时间是以恒定速率递增的,不受系统时间的影响(系统时间可被人为或软件调整)。performance.timing.navigationStart + performance.now() 约等于 Date.now()

let t0 = window.performance.now();
doSomething();
let t1 = window.performance.now();
console.log("doSomething函数执行了" + (t1 - t0) + "毫秒.")

通过这个方法,我们可以用来测试某一段代码执行了多少时间。

performance.mark() mark方法用来自定义添加标记时间。使用方法如下:

  var nameStart = 'markStart';
var nameEnd = 'markEnd';
// 函数执行前做个标记
window.performance.mark(nameStart);
for (var i = 0; i < n; i++) {
doSomething
}
// 函数执行后再做个标记
window.performance.mark(nameEnd);
// 然后测量这个两个标记间的时间距离,并保存起来
var name = 'myMeasure';
window.performance.measure(name, nameStart, nameEnd);

保存后的值可以通过 ** performance.getEntriesByname( 'myMeasure' )**或者 performance.getEntriesByType('measure')查询。

Performance.clearMeasures() 从浏览器的性能输入缓冲区中移除自定义添加的 measure

Performance.getEntriesByName() 返回一个 PerformanceEntry 对象的列表,基于给定的 name 和 entry type

Performance.getEntriesByType() 返回一个 PerformanceEntry 对象的列表,基于给定的 entry type

Performance.measure() 在浏览器的指定 start mark 和 end mark 间的性能输入缓冲区中创建一个指定名称的时间戳,见上例

Performance.toJSON() 是一个 JSON 格式转化器,返回 Performance 对象的 JSON 对象

资源缓冲区监控

Performance.setResourceTimingBufferSize() 设置当前页面可缓存的最大资源数据个数,entryType为resource的资源数据个数。超出时,会清空所有entryType为resource的资源数据。参数为整数(maxSize)。

配合performance.onresourcetimingbufferfull事件可以有效监控资源缓冲区。当entryType为resource的资源数量超出设置值的时候会触发该事件。

Performance.clearResourceTimings() 从浏览器的性能数据缓冲区中移除所有的 entryType 是 "resource" 的 performance entries 下面是mdn上关于这个属性的一个demo。这个demo的主要内容是当缓冲区内容满时,调用buffer_full函数。

function buffer_full(event) {
console.log("WARNING: Resource Timing Buffer is FULL!");
performance.setResourceTimingBufferSize(200);
}
function init() {
// Set a callback if the resource buffer becomes filled
performance.onresourcetimingbufferfull = buffer_full;
}
"init()">

使用performance的这些属性和方法,能够准确的记录下我们想要的时间,再加上日志采集等功能的辅助,我们就能很容易的掌握自己网站的各项性能指标了。

兼容性

目前主流浏览器虽然都已支持performance对象,但是并不能支持它上面的全部属性和方法,有些细微的差别。本文主要依据chrome和qq浏览器测试了相关属性和方法,均可使用。

我们做了什么?(划重点)

现在的很多性能监控分析工具都是通过数据上报来实现的,不能及时有效的反馈页面的性能问题,只能在用户使用之后上报(问题出现之后)才能知道。所以基于新闻前端团队基于performance API做了一款实时查看性能的的工具,它并能给出详细的报表,在开发阶段把性能问题给解决掉。

superProfiler**【外部开源流程中】**

251e349da8ea2cc9494242606c1a7ccb.png

它是一款JavaScript性能监控工具库,通过脚本引用,加载展示在页面右侧,无须依赖任何库和脚本,可以实时查看当前页面的FPS、代码执行耗时、内存占用以及当前页面的网络性能,资源占用。

355c9bb89252ac8d78cc150785524998.png

还能查看最近的(10次)页面性能的平均数。点击“生成报表”按钮会生成更详细的数据报表概览。

525815c8e361a7a601cd30a437902f3d.png

小结

Performance API 用来做前端性能监控非常有用,它提供了很多方便测试我们程序性能的接口。比如mark和measure。很多优秀的框架也用到了这个API进行测试。

它里面就频繁用到了mark和measure来测试程序性能。所以想要开发高性能的web程序,了解Performace API还是非常重要的。最后通过superProfiler工具可以更快更便捷的查找出性能问题,针对性的击破问题,提高开发效率,提升用户体验。当然这只是前端性能优化的第一步,道阻且长。希望大家提出问题和指出疑问,一起进步。

转载自腾讯新闻TNFE前端团队

2f38ec743fd29c155bd35b85b078d77c.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/559353.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java in action,java 7 in action

《JAVA语言程序设计》期末考试试题及答案7_远程、网络教育_成人教育_教育专区。《JAVA语言程序设计》期末考试试题及答案7 一、选择题 1. 请说出下列代码的执行结果......v alidateTree Methods inherited from class javaponent action, add, addCom ponentListener, addFocus…

信号的采样与恢复matlab实验报告,实验七 连续信号的采样与恢复

一、实验目的通过MATLAB仿真验证抽样定理&#xff0c;进一步加深对抽样定理的理解。二、实验原理1. 连续信号的采样对某一连续时间信号f(t)的采样原理图为&#xff1a;由图可知&#xff0c;&#xff0c;其中&#xff0c;单位冲激采样信号的表达式为&#xff1a; 其傅里叶变换为…

treegrid 如何获取getchanges inserted_如何避开Vue性能优化之路的荆棘?

随着这几年前端技术快速发展&#xff0c;Vue框架在国内普及率极高&#xff0c;人人都会用&#xff0c;那Vue如何写得比别人优雅&#xff1f;如何写得比别人漂亮&#xff1f;鉴于一线互联网大厂在前沿技术领域的持续研究和大规模投入&#xff0c;直接向他们取经&#xff0c;是最…

matlab+信号+mpf,Python和Matlab中平均频率的差异

我有这个EMG signal&#xff0c;我想根据这个article绘制平均功率频率。我使用以下代码在Matlab中实现它&#xff1a;clear all;close all;EMGload(EMG.txt);N1000; %my windowz1;fs200 %sampling ratefor i1:length(EMG)-NDUM0;NUM0;FTfft(EMG(i:iN-1));psdFT.*conj(FT);NFFTl…

matlab和vlfeat关联,VLFeat在matlab和vs中安装

转&#xff1a;http://blog.csdn.net/u011718701/article/details/51452011博主最近用vlfeat库做课题&#xff0c;网上搜索使用方法&#xff0c;一大片都会告诉你说&#xff1a;run(/vl_setup) 然后就好啦哈哈哈哈哈哈~~~~~~~~~~~~~~But!理想很丰满&#xff0c;现实很骨感&…

document.createelement如何绑定点击事件_番外篇-EXCEL如何使用宏(VBA)

小编&#xff0c;会在后续推送一些关于VBA(宏)相关的文章(之前其实也推送了一些&#xff0c;其中也大概说了一下如何使用)&#xff0c;所以我们今天就专门写一篇如何使用宏&#xff0c;方便大家使用已经写好的宏。如何使用宏&#xff1a;常规宏VBA-一般都是此类1、调出-开发工具…

mysql --max_allowed_packet=32m,如何修改MySQL-max_allowed_packet

首先查找my.cnf的配置文件(为了让配置永久生效)mysql --help | grep my.cnf修改mysql配置文件&#xff0c;找到vim /etc/my.cnfmax_allowed_packet 1024M >改为2048M保存退出&#xff0c;重新启动mysql服务、/etc/init.d/mysqld restart登录mysql通过本地地址mysql -uroot…

局域网打印机反应慢_为什么你的Excel这么慢,这些原因必须要知道!

Excel是很多人的高频办公工具&#xff0c;但大家或多或少地遇到过&#xff0c;有时自己电脑的Excel很慢&#xff0c;导致工作的效率不高&#xff0c;可能就会导致要加班加点&#xff0c;也压缩了自己其他方面的时间。本文将介绍一些常见的Excel”慢“的情况及其解决方案&#x…

php密码安全检测,php – 密码安全随机字符串函数

目标&#xff1a;找到最加密的安全随机字符串生成器.在字符串中使用字母,数字和可能的特殊字符.我一直在这里和其他地方读书,但我仍然听到很多不同的答案/意见.那些对安全性和加密技术有最新知识且知识渊博的人可以在这里发出声音.以下函数将用于生成8个字符的随机密码,并生成1…

linux挂载fc存储有超级坏块_Nand Flash基础知识与坏块管理机制的研究

欢迎FPGA工程师加入官方微信技术群点击蓝字关注我们FPGA之家-中国最好最大的FPGA纯工程师社群概述Flash名称的由来&#xff0c;Flash的擦除操作是以block块为单位的&#xff0c;与此相对应的是其他很多存储设备&#xff0c;是以bit位为最小读取/写入的单位&#xff0c;Flash是一…

php 设置agent,限制某个目录禁止解析php及user_agent、php相关配置

一、 限定某个目录禁止解析php1、 核心配置文件内容php_admin_flag engine off2、curl测试时直接返回了php源代码&#xff0c;并未解析二、 限制user_agent1、 user_agent可以理解为浏览器标识2、核心配置文件内容RewriteEngine onRewriteCond %{HTTP_USER_AGENT} .curl. [NC,O…

python 打印xml文档树_[Python]xml.etree.ElementTree处理xml文档

需求&#xff1a;在实际应用中&#xff0c;需要对xml配置文件进行实时修改&#xff0c;1.增加、删除 某些节点2.增加&#xff0c;删除&#xff0c;修改某个节点下的某些属性3.增加&#xff0c;删除&#xff0c;修改某些节点的文本xml源文件格式[例]path"/tmp">out…

oracle最快访问行,Oracle技术网—在Oracle快速进行数据行存在性检查

在Oracle快速进行数据行存在性检查当在应用程序的业务逻辑中需要检查一个外键是否有相关的主键时&#xff0c;往往使用Select Count(*)类型的SQL语句。这是一个很显而易的方法&#xff0c;但却不是最快的方法。Count(*)函数调用可能会引起对整个表的进行扫描&#xff0c;这是一…

carsim中质心加速度_振动CAE分析在空调压缩机支架设计中的应用

【摘要】本文运用有限元分析方法分析空调压缩机系统模态&#xff0c;并通过分析引起振动的激励源&#xff0c;找出压缩机支架和安装螺栓断裂的根本原因&#xff0c;并根据分析提出了解决措施。关键词&#xff1a;空调压缩机支架模态激励共振一、引言发动机轮系需要驱动的工作部…

oracle windows server 2008,Node.js 在 Windows Server 2008 X64 连接Oracle 数据库

不需要安装Oralce客户端&#xff0c;不影响其它使用Oracle客户端的程序运行安装Node.JSnode-v12.13.0-x64.msi安装VC 2013运行时vcredist_x64.exe运行的版本要与Oracle 客户端匹配&#xff0c;下载Oracle 客户端instantclient-basiclite-windows.x64-18.5.0.0.0dbru.zip解压并放…

python中end 1 意思_python中“end=”是什么意思?_后端开发

c语言fopen函数的用法_后端开发在c语言中可以使用“fopen()”函数打开一个文件&#xff0c;该函数用于打开一个文件&#xff0c;其语法是“open(“file a”,”r”);”&#xff0c;其参数r表示只读属性。python中“end”是“print()”函数中的一个参数&#xff0c;会使该函数关闭…

oracle如何收缩表空间,ORACLE收缩表空间

最近导一个空库到数据库后&#xff0c;发现占用的表空间非常大&#xff0c;执行表收缩(SHRINK SPACE CASCADE)后&#xff0c;发现实际占用的空间不到1%。 但是收缩表空间大小提示错误 ALTER DATABASE DATAFILE D:\ora_tablespace\GCOMM2.dbf RESIZE 5000M; 提示&#x…

python 类 对象 知乎_python基础知识:类,对象,模块三者的区别

长话短说&#xff0c;我线代作业还没有写呢。有点编程语言知识的朋友应该知道&#xff0c;编程语言有“面向对象”和“面向过程”两种。解释一下&#xff1a;面向对象技术是目前流行的系统设计开发技术&#xff0c;它包括面向对象分析和面向对象程序设计。面向对象程序设计技术…

linux下usb驱动配置文件,Linux环境下USB的原理、驱动和配置

随着生活水平的提高&#xff0c;人们对USB设备的使用也越来越多&#xff0c;鉴于Linux在硬件配置上尚不能全部即插即用&#xff0c;因此关于Linux如何配置和使用&#xff0c;成为困扰我们的一大问题。本文引用地址&#xff1a;http://www.eepw.com.cn/article/70029.htmlinux操…

python的数据库中间件_数据库中间件设计方案

数据库中间件的主要作用是向应用程序开发人员屏蔽读写分离和分库分表面临的挑战&#xff0c;并隐藏底层实现细节&#xff0c;使得开发人员可以像操作单库单表那样去操作数据。在介绍分库分表的主流设计方案前&#xff0c;我们首先回顾一下在单个库的情况下&#xff0c;应用的架…