[译]从输入URL到页面呈现的超详细过程——第二步:Tags转化成DOM的过程

栏目: 后端 · 前端 · 发布时间: 5年前

内容简介:这是一个系列文章,分为四个部分介绍了从输入URL到页面呈现的详细过程本篇翻译的是第二部分:Tags to DOM在上一篇文章中(

原文链接:Tags to DOM

原文作者:Travis Leithead

译者:wangds

这是一个系列文章,分为四个部分介绍了从输入URL到页面呈现的详细过程

  • 客户端从服务器获取资源(Server to Client) 点击查看
  • 标签转化成DOM的过程(tags to DOM)
  • CSS解析的过程(braces to pixels)
  • 编译执行javascript的过程(var to JIT)

本篇翻译的是第二部分:Tags to DOM

在上一篇文章中( 客户端从服务器获取资源 ),我们谈论了资源是如何从服务器到达客户端的,同时也阐述了一些关于缓存、同源的概念。本篇文章将聊一聊 HTML 资源是如何转化成 DOM tree 的。

解析的过程(Parsing)

当浏览器获得了资源以后要进行的第一步工作就是解析,整个解析的过程可以拆分成如下四个步骤:

encoding
pre-parsing
Tokenization
tree construction

1.解码( encoding )

客户端接收的内容可以是 HTML 文本、图像等等任何格式的数据。在从服务器到客户端展示的过程中,数据经过了如下的过程:

bit
bit
bit

所以客户端必须知道数据转化成 bit 的格式,以便客户端将 bit 反转成正确的数据。

我们以 HTML 举例来说, HTML 的文本格式有很多种,所以服务器在返回数据的时候在响应头中使用 Content-Type 告知浏览器文本的内容类型,同时在文本文件头部使用 Byte Order Mark 告知浏览器编码格式。如果浏览器仍然判断不了解码方式,浏览器还会自行匹配一种解码格式来处理数据。有时候,解码格式也会写在 <meta> 标签中。这就导致会出现一种奇葩的现象,浏览器先是以自己的方式去解码文本,然后遇到了标明解码方式的 <meta> 标签。这时候浏览器就会重新解析按照 <meta> 标签进行解析。

我们现在经常在 HTML 中使用的文件格式是 UTF-8 ,那是因为 UTF-8 能较完整的支持 Unicode 字符范围,同时与 CSSJavaScript 中常见的节字符具有良好的 ASCII 兼容性。一般浏览器默认的解码格式也是 UTF-8 。当解码出错的时候,我们会看到屏幕上全部都是乱码字符。

2.预解析(Pre-parsing/scanning)

完成了解码以后,浏览器会开启一个预解析的过程,“偷偷获取”后面需要加载的资源,减少处理的时间。预解析的过程不是一个完整的解析过程。举例来说,预解析不会处理 HTML 内元素之间嵌套的父子层级。但是预解析还是能识别一些特殊的 HTML 标签、元素属性值以及 URL 。假如我们的页面中有个 img (如下),预解析就会注意 src 属性值,并将获取这个图片的请求加到请求队列中,尽早的将图片获取到本地以便显示。我们也可以通过 prefetchpreload 两种属性将需要尽早加载的资源告知于浏览器。

<img src="https://somewhere.example.com/​images/​dog.png" alt="">
复制代码

3.符号化(TOKENIZATION)

符号化是将输入解析成为符号,像开始标签( < )、结束标签( > )、注释( // )、文本内容等等。然后这些符号将被送到解析的下一个步骤。完成符号化工作的机器叫做符号识别器( tokenizer ),符号识别器是一种状态机。 我们以 <video controls> 的标签为例:

  • 状态机初始状态是 Data State ,
  • 当遇到 < 的时候,状态变成 Tag open state
  • 当解析到字母v的时候,状态变成 Tag name state
  • 当解析到字母c的时候,状态变成 in attribute name state
  • 解析完 字母s的时候,状态变成 after attribute name state
  • 解析完 > 的时候,状态变成 Data state

从解析 <video controls> 标签我们就可以知道,整个页面会碰到很多标签,符号识别器就会不断的重复状态变化。(下图中的箭头就是状态机处于的某个位置)

[译]从输入URL到页面呈现的超详细过程——第二步:Tags转化成DOM的过程

HTML标准 目前为符号识别器定义了80个独立的状态。在解析的过程中,符号识别器可以将任何的文本转化成 HTML 文档(即便文本内容不是有效的 HTML )。这种弹性的机制使得 HTML 更加的易于开发,但是也可能导致因为写错标签出现特殊的bug。

对于那些更喜欢使用‘非黑即白’验证规则的人,浏览器内置了一种替代解析机制,可将任何解析错误视为灾难性的故障(出现错误就会导致无法呈现内容)。 这种解析模式使用 XML 规则来处理 HTML ,我们可以通过设置 MIME 类型的值为 application / xhtml + xml 开启这种解析模式。

浏览器可以同时处理预解析和标记化两个过程(也就是说这是两个线程)。

4. 构建树( tree construction )

在上一步符号化以后,解析器获得这些符号标记,然后以合适的方法创建 DOM 对象并将这些符号插入到 DOM 对象中。 DOM 对象的数据结构是树状的,所以这个过程称为构造树( tree construction )。顺便说一句,在 IE 的历史中,大部分时间里没有使用树结构。

[译]从输入URL到页面呈现的超详细过程——第二步:Tags转化成DOM的过程

另外现代浏览器为了兼容旧版本 HTML ,解析的过程还是十分复杂的。例如浏览器可以自动闭合标签。

<p>sincerely<p>The authors</p> //HTML的样子

<p>sincerely</p><p>The authors</p>//浏览器解析出来的样子
复制代码

我们可以使用 JavaScript 去任意修改 DOM 。假如我们用 JavaScript 进行无意义的操作(例如在 video 元素内插入一个 table 单元格),渲染引擎可以甄别出这些操作不合规,也不进行渲染(但是这个过程渲染引擎也进行工作只是不展示)。

因为 JavaScript 可以在 DOM 中添加内容。所以遇到 JavaScript 文件的时候, DOM 将停止解析;如果 JavaScript 文件内调用了 document.write API,解析器将重新开始解析过程。

事件(Events)

当整个解析的过程完成以后,浏览器会通过 DOMContentLoaded 事件来通知 DOM 解析完成。事件相当于一套广播系统用来通知和监听浏览器。除了 DOMContentLoaded 事件,还有 load 事件(表示所有资源已经加载完成,包括图片、视频、 CSS 等等)、 unload 事件表示界面即将关闭、鼠标事件键盘事件等等。

浏览器在 DOM 对象中还会创建一个事件对象,该对象包含大量有用的信息(如屏幕中的坐标、按下的按键等等)。当 JavaScript 触发事件的时候,就会同时产生事件对象。

在目标元素上触发事件的时候,需要从 DOM 树的根元素开始向子元素查找,这个过程俗称事件捕捉阶段。到达目标元素以后,还要逐级向上返回到根元素上,这个过程俗称事件冒泡阶段。

[译]从输入URL到页面呈现的超详细过程——第二步:Tags转化成DOM的过程

事件是可以进行关闭的,我们可以在事件捕获或者传播的过程中选择关闭,例如在 form 的提交过程中判断字符串为空则关闭提交。

DOM

解析器解析出来的元素可以包含一些基本的交互功能( <video> 能播放视频、 <button> 按钮等),但是这些基本的交互功能还不能满足我们的需求,我们需要使用 cssJavaScript 将页面展示的更加丰富多彩,而 DOM 就是 HTML 元素与其他资源交互的桥梁。

元素的接口

在解析器将元素放入 DOM 树之前,解析器会根据不同元素的名称赋予元素不同的接口功能。

下面是一些通用的功能:

DOM

<table> 元素具有查找删除表内所有单元格的能力; <canvas> 具有描绘线条、图形的能力。但是想要调用这个接口还是需要 JavaScript 来操作。

每当我们使用 JavaScript 操作 DOM 的时候,将会触发浏览器的一些连锁反应,这些反应是为了让更改后的页面更快的呈现在屏幕上。例如:

  • 用数字代表通用的元素名称和属性,浏览器用使用哈希表进行快速识别这些数字
  • 将频繁变更的子元素进行缓存,方便子元素快速迭代
  • sub-tree 的跟踪变化降到最低,避免‘污染’整个 DOM

DOM 中的 HTML 元素是页面展示的基础, DOM 的能力也不仅仅局限于上面我们提到的功能。其他诸如访问储存系统(浏览器中的缓存)、设备功能(蓝牙、定位)、以及3D图形绘制等等功能。随着浏览器的不断进步和web标准的不断实施, DOM 的功能只会越来越强大,本文只涉及了一部分而已。


以上所述就是小编给大家介绍的《[译]从输入URL到页面呈现的超详细过程——第二步:Tags转化成DOM的过程》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

The Algorithm Design Manual

The Algorithm Design Manual

Steven S Skiena / Springer / 2011-11-14 / GBP 55.07

....The most comprehensive guide to designing practical and efficient algorithms.... Written by a well-known algorithms researcher who received the IEEE Computer Science and Engineering Teaching Aw......一起来看看 《The Algorithm Design Manual》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

随机密码生成器
随机密码生成器

多种字符组合密码

URL 编码/解码
URL 编码/解码

URL 编码/解码