最近跟公司同事分享session时在谈到CSS优先级权重问题,引入了之前一直在网上公认的权重计算公式,有好奇者问了一个ClASS选择器组合与ID选择器优先级问题,如果按公式套下来,CLASS组合后是应该高于ID的,但结果实验并非如此,好奇之下,又在晚上找了找,总结如下。
网上教程权重计算
关于CSS权重,网上教程大都按不同选择器给了权重值:
- STYLE:1000
- #ID:100
- .CLASS 10
- ELEMENT 1
这里引入一张流行的CSS权重关系图:
组合比较
256个class选择器可以干掉1个id选择器
这里引自张鑫旭的一片博客。大致内容是说
Firefox等浏览器所有的类名(classes)都是以8字节字符串存储的,8字节所能hold的最大值就是255. 所以你想啊,当同时出现256个class, 势必会越过其边缘,溢出到id区域。
Opera浏览器class类名的存储是以16字节的字符串。因此,该浏览器要想发生class溢出到id的话,需要连续65536个class。
并且实际做了一个例子来进行比较,代码如下(原文请点击这里):
css:
示例地址http://www.zhangxinxu.com/study/201208/256-class-fire-an-id.html
经测试,在Firefox下的确是CLASS优先级胜过了ID,但在Chrome下依旧是#ID优先级高。
当然,实际开发中谁也不会写这么多得css,也更没法用这种方式来复写id样式。于是又找到下面这篇。
选择器权重值的计算
-
A:如果规则是写在标签的style属性中(内联样式),则A=1,否则,A=0. 对于内联样式,由于没有选择器,所以B、C、D的值都为0,即A=1, B=0, C=0, D=0(简写为1,0,0,0,下同)。
-
B:计算该选择器中ID的数量。如果有则B=1,没有B=0(例如,#header 这样的选择器,计算为0, 1, 0, 0)。
-
C:计算该选择器中伪类及其它属性的数量(包括class、属性选择器等,不包括伪元素)。(例如, .logo[id='site-logo']
这样的选择器,计算为0, 0, 2, 0)(后面将进一解释为什么会是0,0,2,0)。
-
D:计算该选择器中伪元素及标签的数量。(例如,p:first-letter
这样的选择器,计算为0, 0, 0, 2)。
CSS2规范中给出的一些例子:
这种说法是错误的:
根据这样的定义,所以很多文章简单地把规则归纳为:内联样式的权重值是1000,ID选择器的权重值是100,class选择器的权重值是10,标签选择器的权重值是1. 整条规则中的所有选择器权重值相加得到整个样式规则的权重值,数字越大权重值越高。
大多数情况下,按照这样的理解得出的结论没有问题,但是遇到下面这样的情况就出现问题了:
样式一:body header div nav ul li div p a span em {color: red}
样式二:.count {color: blue}
按照错误的计算方法,样式一的权重值是11,样式二的权重值是10,如果这两条规则用于同一个元素,则该元素应该是红色。实际结果却是蓝色。
权重值的比较
按照四组计算的正确方法,上面例子中的样式一权重值应该是0, 0, 0, 11,样式二的权重值是0, 0, 1, 0。
根据规范,计算权重值时,A,B,C,D四组值,从左到右,分组比较,如果A相同,比较B,如果B相同,比较C,如果C相同,比较D,如果D相同,后定义的优先。
样式二和样式一的A、B相同,而样式二的C大于样式一,所以,不管D的值如何,样式二权重值都大于样式一。
这就是正确的答案。
特殊的 !important
CSS2规范中规定:!important 用于单独指定某条样式中的单个属性。对于被指定的属性,有 !important 指定的权重值大于所有未用 !important 指定的规则。
例如:
样式一: #header nav ul li.current {color: red; font-weight: bold;}
样式二: li:hover {color: blue !important; font-weight: normal;}
就整条规则而言,样式一的权重值为 0, 1, 1, 3,而样式二的权重值仅为0, 0, 2, 0。所以应用于相同元素时,应该样式一生效。但是对于color这个属性,由于在样式二中用 !important 做了指定,因此color将应用样式二的规则。而font-weight则按照规定用样式一的规则。
如果多条规则中都对同一个属性指定了 !important 呢?这时候 !important 的作用相互抵销,依然按照ABCD四组计算比较。
因此 !important 的作用只有在具有唯一性时才能提现,但是我们永远无法预料自己什么时候又需要覆盖一个已经指定了 !important 的属性,所以最好的办法就是:不要使用 !important。