关于css:CSS元素选择器是怎样运作的

30次阅读

共计 2569 个字符,预计需要花费 7 分钟才能阅读完成。

在前端工程师的日常工作中,应用 CSS 元素选择器是稀松平时的事;无论你是编写个别的 CSS 还是须要通过编译的 SASS,SCSS,LESS 等,最终都被编译成一行一行的 CSS 款式属性,最终交给浏览器解析并套用。然而你想过没有这是如何实现的呢?

浏览器渲染

咱们先看一下浏览器的渲染步骤:

CSS 在被浏览器加载后,会被解析成 CSSOM 树,并尝试与 Dom 叠加成渲染树,随后进行计算地位、渲染等步骤。这样看来,CSS 属性套用的要害就在于如何从 CSS 转化成 CSSOM 树,以及怎么把 CSSOM 套用到 DOM 下来。

CSSOM 树

当咱们写下一组 CSS 款式时,例如:

#id .class h4 + p {...}

浏览器在解析它时,你可能会认为 CSS 会依照由左到右的依序找出 #id>.class>h4>p,最初套用,但实际上 浏览器解析 CSS 的程序是由右到左 p>h4>.class>#id

很违反直觉对吧?但如果思考到性能问题,从右到左的解析会比从左到右强很多。

假如这有这样的 HTML:

<div id="div1">
    <div class="a">
        <div class="b">
            ...
        </div>
        <div class="c">
            <div class="d">
                ...
            </div>
            <div class="e">
                ...
            </div>
        </div>
    </div>
    <div class="f">
        <div class="c">
            <div class="d">
                ...
            </div>
        </div>
    </div>
</div>

以及这边五条 CSS 款式规定:

#div1 .c .d {}
.f .c .d {}
.a .c .e {}
#div1 .f {}
.c .d {}

让咱们模仿一下,如果把 CSS 从左到右解析,将会生成相似这样的 CSSOM 树:

通过 <div class =“d”> 中的 .d 来思考,这样的 CSSOM 树在套用款式时,必须对 所有 的款式规定进行查看,以确认款式规定是否会影响到 .d,到最初能力确定可能会影响到 .d 的款式规定有这三条:

  • #div1 .c .d
  • .f .c .d
  • .c .d

以此类推,每个 DOM 树上的元素,都必须便当所有的款式规定,才能够获得个别的款式,这样会造成大量冗余的计算,进而重大影响性能。

反过来,如果将后面的 CSS 由右到左进行解析,CSSOM 树则可能会如下:

和后面的例子一样,从 <div class =“d”>.d 的角度来看,因为会被款式规定影响到的指标元素,曾经全都集中在第一层了,所以就不必再去便当整个 CSSOM 树了,甚至只须要查看 .d 以下的子属性变量是否符合实际 DOM 构造,再将所有合乎的款式规定从新取回,便能实现 .d 对元素的款式规定套用。

从右到左的解析程序可能将所有共享的规定门路收拢在一起,当浏览器进行属性比对时,就不必再便当整个 CSSOM 树,大大的缩小了有效的比对计算。

也能够换个形式思考:在 HTML 的构造中,一个元素能够有无数个子元素,但只能有一个父元素,由子找父(由下往上)搜查相对是比拟快的。

套用款式

将 CSSOM 树解析进去之后就可能和 DOM 联合了吗?如果真的有这么简略就太好了。

除了开发者定义好的 CSS 档外,还有几个中央可能会定义款式规定,影响画面的渲染:

  • HTML 的 inline style 设置
  • 浏览器预设值(就是 CSS reset/normalize 要笼罩掉的货色)
  • 浏览器的使用者偏好设定

浏览器负责解决 CSS 的局部,会吧后面所有的货色以及 CSS 文件定义的款式规定别离整顿成独自的款式规定组(CSS 规定集),内容记录了款式规定、指标属性等信息。

指标属性

为了晋升前面的计算效率,浏览器的 CSS 解决内核会依照款式规定组中个别规定的指标属性将其分组寄存;一共分为以下四组

  • idRules
  • classRules
  • tagNameRules
  • universalRules

这样在取用时,能够根据指标元素是否存在这个属性,疾速筛出可能会套用的款式。

套用规定

最初是套用规定。浏览器会遵循以下程序和款式规定权重套用所有的款式规定:

  • 浏览器的预设值
  • 浏览器的使用者偏好设定
  • 开发者定义的 CSS
  • inline style
  • 加上 !important 的款式属性

你可能会好奇:为什么 inline style 和开发者定义的 CSS 会被另外解决?

咱们能够回顾一下浏览器渲染的步骤,因为 inline style 存在于 DOM 元素中,只能在 CSS 套用到 DOM 上时才会接触到,事先无奈将两者联合。

CSS 效率

实际上浏览器在这里曾经实现了优化机制;浏览器会主动将状态统一的元素做款式快照。状态统一就是要满足以下几个条件:

  • 没有设定 ID
  • tag 及 class 必须完全一致
  • 没有设定 style 属性
  • 款式规定中不能应用各种同级选择器(例如:+:first-child 等)

因为下面的条件,以及后面探讨到的 CSS 运算过程,编写 CSS 时也有几个中央能够略微留心一下:

  • 因为款式规定的指标属性会分组寄存,id 选择器效率十分高,所以是不能与其余条件混用的。
  • 不要写过深的 CSS 款式规定
  • 能不必 inline style 就不要用,除了难以保护外,因为是存在于 DOM 树上,无奈事后与其余款式合并计算,所以效率也会大打折扣

如果可能留神到这类典型的小细节,CSS 效率天然也能够大幅晋升。

延长

意识了 CSS 选择器之后,你肯定会很好奇,JavaScript 的元素选择器又是怎么回事呢?这个问题能够参考 jQuery 的源码,它是由左到右的解析,至为什么为什么不一样,其实在文中也有答案,就留给你思考开掘吧。


本文首发微信公众号:前端先锋

欢送扫描二维码关注公众号,每天都给你推送陈腐的前端技术文章

欢送持续浏览本专栏其它高赞文章:

  • 深刻了解 Shadow DOM v1
  • 一步步教你用 WebVR 实现虚拟现实游戏
  • 13 个帮你进步开发效率的古代 CSS 框架
  • 疾速上手 BootstrapVue
  • JavaScript 引擎是如何工作的?从调用栈到 Promise 你须要晓得的所有
  • WebSocket 实战:在 Node 和 React 之间进行实时通信
  • 对于 Git 的 20 个面试题
  • 深刻解析 Node.js 的 console.log
  • Node.js 到底是什么?
  • 30 分钟用 Node.js 构建一个 API 服务器
  • Javascript 的对象拷贝
  • 程序员 30 岁前月薪达不到 30K,该何去何从
  • 14 个最好的 JavaScript 数据可视化库
  • 8 个给前端的顶级 VS Code 扩大插件
  • Node.js 多线程齐全指南
  • 把 HTML 转成 PDF 的 4 个计划及实现

  • 更多文章 …

正文完
 0