对网页渲染的初步认识

老实说,这篇博客是看了园子里的OneAPM官方技术博客关于有关网页渲染,每个前端开发者都该知道的那点事来写的,真心觉得不错,对于想做前段开发的程序员来说,是不可不知知的知识。

浏览器是如何完成网页渲染的?


网页渲染时,浏览器的动作有:

  1. 根据来自服务器的html代码形成文档对象模型(DOM)
  2. 加载并解析样式,形成CSS对象
  3. 在文档对象模型和CSS对象模型之上,创建一棵由一组待生成渲染的对象组成的渲染树(在Webkit(典型代表是Google Chrome)中这些对象被称为渲染器或渲染对象,而在Gecko(典型代表是Firefox)中称之为“frame”。)渲染树反映了文档对象模型的结构,但是不包含诸如标签或含有display:none属性的不可见元素。在渲染树中,每一段文本字符串都表现为独立的渲染器。每一个渲染对象都包含与之对应的DOM对象,或者文本块,还加上计算过的样式。换言之,渲染树是一个文档对象模型的直观展示
  4. 对渲染树上的的每一个元素,计算它的坐标,称之为布局。浏览器采用一种流方法,布局一个元素只需通过一次,但是表格元素需要通过多次。
  5. 最后,渲染树上的元素最终展示在浏览器里,这一过程称为“painting”

当用户与网页交互,或者脚本程序改动修改网页时,前文提到的一些操作将会重复执行,因为网页的内在结构已经发生了改变。

重绘


  • 当改变那些不会影响元素在网页中的位置的元素样式时,譬如background-color(背景色), border-color(边框色), visibility(可见性),浏览器只会用新的样式将元素重绘一次(这就是重绘,或者说重新构造样式)

重排


当改变影响到文本内容或结构,或者元素位置时,重排或者说重新布局就会发生。那产生重排的条件有哪些呢?通常有以下几类:

  1. DOM操作(元素添加,删除,修改,或者元素顺序的改变);
  2. 内容变化,包括表单域内的文本改变;
  3. CSS属性的计算或改变;
  4. 添加或删除样式表;
  5. 更改“类”的属性;
  6. 浏览器窗口的操作(缩放,滚动);
  7. 伪类激活(:悬停)。

浏览器如何优化渲染?


  1. 浏览器尽可能将重绘/重构 限制在被改变元素的区域内。比如,对于位置固定或绝对的元素,其大小改变只影响元素本身及其子元素,然而,静态定位元素的大小改变会触发后续所有元素的重流。

  2. 另一种优化技巧是,在运行几段JavaScript代码时,浏览器会缓存这些改变,在代码运行完毕后再将这些改变经一次通过加以应用。举个例子,下面这段代码只会触发一个重构和重绘:

    var $body = $(‘body‘);
    $body.css(‘padding‘, ‘1px‘); // reflow, repaint
    $body.css(‘color‘, ‘red‘); // repaint
    $body.css(‘margin‘, ‘2px‘); // reflow, repaint
    // only 1 reflow and repaint will actually happen
  3. 然而,如前所述,改变元素的属性会触发强制性的重排。如果我们在上面的代码块中加入一行代码,用来访问元素的属性,就会发生这种现象。

    var $body = $(‘body‘);
    $body.css(‘padding‘, ‘1px‘);
    $body.css(‘padding‘); // reading a property, a forced     reflow
    $body.css(‘color‘, ‘red‘);
    $body.css(‘margin‘, ‘2px‘);
  4. 其结果就是,重排发生了两次。因此,你应该把访问元素属性的操作都组织在一起,从而优化网页性能。(你可以在JSBin查到更为详细的例子

    有时,你必须触发一个强制性重排。比如,我们必须将同样的属性(比如左边距)两次赋值给同一个元素。起初,它应该设置为100px,且不带动效。接着,它必须通过过渡(transition)动效改变为50px。你现在可以在JSbin上学习这个例子,不过我会在这儿更详细地介绍它。

    首先,我们创建一个带过渡效果的CSS类:

    .has-transition {
        -webkit-transition: margin-left 1s ease-out;
    -moz-transition: margin-left 1s ease-out;
    -o-transition: margin-left 1s ease-out;
    transition: margin-left 1s ease-out;
    }

  5. 然后继续执行:

    // our element that has a "has-transition" class by default
    var $targetElem = $(‘#targetElemId‘);
    
    // remove the transition class
    $targetElem.removeClass(‘has-transition‘);
    
    // change the property expecting the transition to be     off, as the class is not there
    // anymore
    $targetElem.css(‘margin-left‘, 100);
    
    // put the transition class back
    $targetElem.addClass(‘has-transition‘);
    
    // change the property
    $targetElem.css(‘margin-left‘, 50);

  6. 然而,这个执行无法奏效。所有改变都被缓存,只在代码块末尾加以执行。我们需要的是强制性的重排,我们可以通过以下更改加以实现:

    // remove the transition class
    $(this).removeClass(‘has-transition‘);
    
    // change the property
    $(this).css(‘margin-left‘, 100);
    
    // trigger a forced reflow, so that changes in a     class/property get applied immediately
    $(this)[0].offsetHeight; // an example, other     properties would work, too
    
    // put the transition class back
    $(this).addClass(‘has-transition‘);
    
    // change the property
    $(this).css(‘margin-left‘, 50);

有关性能优化的实际建议


  1. 创建有效的HTML和CSS文件,不要忘记指明文档的编码方式。样式应该包含在标签内,脚本代码则应该加在标签末端。
  2. 尽量简化和优化CSS选择器(这种优化方式几乎被使用CSS预处理器的开发者统一忽视了)将嵌套程度保持在最低水平。以下是CSS选择器的性能排名(从最快者开始):
    1. 识别器:#id
    2. 类:.class
    3. 标签:div
    4. 相邻兄弟选择器:a + i
    5. 父类选择器:ul> li
    6. 通用选择器:*
    7. 属性选择:input[type="text"]
    8. 伪类和伪元素:a:hover
  3. 你应该记住,浏览器在处理选择器时依照从右到左的原则,因此最右端的选择器应该是最快的:#id或则.class:

    div * {...} // bad
    .list li {...} // bad
    .list-item {...} // good
    #list .list-item {...} // good
    1. 在你的脚本代码中,尽可能减少DOM操作。缓存所有东西,包括元素属性以及对象(如果它们被重用的话)。当进行复杂的操作时,使用“孤立”元素会更好,之后可以将其加到DOM中(所谓“孤立”元素是与DOM脱离,仅保存在内存中的元素)。
    2. 如果你使用jQuery来选择元素,请遵从jQuery选择器最佳实践方案。
    3. 为了改变元素的样式,修改“类”的属性是奏效的方法之一。执行这一改变时,处在DOM渲染树的位置越深越好(这还有助于将逻辑与表象脱离)。
    4. 尽量只给位置绝对或者固定的元素添加动画效果。
    5. 在使用滚动时禁用复杂的悬停动效(比如,在中添加一个额外的不悬停类)

原文地址:http://www.cnblogs.com/doctorJoe/p/4698750.html

时间: 2024-10-14 20:46:49

对网页渲染的初步认识的相关文章

有关网页渲染,每个前端开发者都该知道的那点事

[编者按]其实,有关网页渲染的文章很多,但是相关信息比较分散,且论述并不是很完整.如果要想对这个主题有个大致的了解,我们还得学习很多知识.因此,Web开发者Alexander Skutin 决定写一篇文章.他相信,这篇文章不仅能帮助初学者,也能对那些想要刷新知识结构的高级前端开发者有所裨益.原文地址 译文如下: 网页渲染必须在很早的阶段进行,可以早到页面布局刚刚定型.因为样式和脚本都会对网页渲染产生关键性的影响.所以专业开发者必须了解一些技巧,从而避免在实践的过程中遇到性能问题. 这篇文章不会研

webkit技术--网页渲染原理

Webkit渲染 Webkit 是苹果发起的一个开源项目,后来谷歌用这个项目以 webkit 创建了一个新的项目 Chromium,我们平常用的 Chrome 浏览器一般都是基于 Chromium 开发出来的稳定版本.本文主要介绍浏览器的渲染引擎即 webkit内核.webkit 作为浏览器的渲染引擎能够将 html/css/js文本及相应的资源文件转换成图像结果. 1.内部模块结构图如下: 可以看出渲染引擎大致包括HTML解释器,CSS解释器,布局和JavaScript引擎等模块.这些模块可能

Chromium网页渲染机制简要介绍和学习计划

作为一个浏览器,快速地将网页渲染出来是最重要的工作.Chromium为了做到这一点,费尽了心机,做了大量优化工作.这些优化工作是卓有成效的,代表了当今最先进的网页渲染技术.值得一提的是,这些渲染技术不仅适用于网页渲染,也可以应用在原生系统的UI渲染上.例如,在Android系统上,我们就可以看到两者在渲染技术上的相似之处.本文接下来就对Chromium的网页渲染机制进行简要介绍,并且制定学习计划. 老罗的新浪微博:http://weibo.com/shengyangluo,欢迎关注! Chrom

Chromium网页渲染调度器(Scheduler)实现分析

在采用线程化渲染方式渲染网页时,Chromium依赖一个调度器协调Main线程和Compositor线程的执行,同时也通过这个调度器决定它们什么时候该执行什么操作.调度器将Main线程和Compositor线程的当前状态记录在一个状态机中,然后通过这个状态机决定下一个要执行的操作.这个操作在满足当前设置条件下是最优的,因此可以使网页渲染更快更流畅.本文接下来就分析Chromium网页调度器的实现. 老罗的新浪微博:http://weibo.com/shengyangluo,欢迎关注! 调度器实现

影响网页渲染的关键!

经常有站长.开发者.运维疑惑:为什么我们的后台服务器很快,但是用户要看网页里面的内容却需要很长时间?我们在上一篇文章<怪兽大作战: 解析网站打开慢的原因>中简单介绍了影响网站打开速度的几个指标,感兴趣的同学可以再读一下.今天我们主要讲一下,是哪些因素拖慢了我们的首屏加载时间,也就是用户看到网页中内容时所等待的时间. 用过OneAPM的读者对这幅图肯定不陌生,一般来讲,如果服务器很快,机房所在线路很快,那么影响用户看到网页内容的主要时间,就是最后两个时间阶段:DOM处理以及网页渲染,在这两个阶段

影响网页渲染的关键

经常有站长.开发者.运维疑惑:为什么我们的后台服务器很快,但是用户要看网页里面的内容却需要很长时间?我们在上一篇文章<怪兽大作战: 解析网站打开慢的原因>中简单介绍了影响网站打开速度的几个指标,感兴趣的同学可以再读一下.今天我们主要讲一下,是哪些因素拖慢了我们的首屏加载时间,也就是用户看到网页中内容时所等待的时间. 用过OneAPM的读者对这幅图肯定不陌生,一般来讲,如果服务器很快,机房所在线路很快,那么影响用户看到网页内容的主要时间,就是最后两个时间阶段:DOM处理以及网页渲染,在这两个阶段

前端开发工程师应知应会之网页渲染(翻译)

作者:Alexander Skutin , 2014.5.26 . 由Max shirshin与2014年6月30日翻译(俄语 -> 英语) 现今我们应更加注重网页渲染,及其在web开发中的重要性.虽然很多文章都曾谈到这一主题,但大多是分散和割裂的.譬如为了对这个主题有更全面的认识需要去搜索很多的信息来源,而这也是笔者决定写这篇文章的原因.笔者相信本篇文章会有益于初级开发者,当然对希望能够更新和整理已有知识的中高级开发者同样能够有所裨益. 当页面布局定义完成后,页面渲染的过程与样式和脚本所承担的

由网页渲染提出的优化建议

总所周知,网页加载渲染分为一下几个阶段: -构建dom树(dom tree) 从上到下解析html构建dom树,也叫内容树 -构建css树(CSSOM) 将css样式附着到dom树上生成CSSOM tree(css object model tree) -执行js 执行js代码(同步) -构建渲染树(rebder tree) 渲染树代表一个稳定的视觉呈现,有一系列矩形构成,带有大小.颜色.字体等属性, 这一个个小矩形,firefox称为框架(frame),webkit称为渲染对象(render

网页渲染优化和JS性能提升

自学前端有一段时间了,在网上学习总结了点网页的渲染性能和JS性能提升的方法,有利于写出更加优秀的代码 1. 不要使用GIF图片实现loading效果,这样会降低CPU消耗: 2. 禁止使用iframe,会阻塞父文档onload事件:搜索引擎的检索程序无法解读这种页面,不利于SEO: 3. 使用CSS3代码替代JS动画,尽可能避免重绘重排以及回流: 4. 对于一些小图标,可以使用base64位编码,以减少网络请求: 5. 网页Gzip,CDN托管,data缓存 ,图片服务器: 6. 前端模板 JS