跳至主要內容

CSS的单位

yzqdev大约 5 分钟

CSS的单位

inherit

大多数 CSS 属性默认继承它的父元素的设定,但是有一些属性默认不继承。比如,input元素和textarea元素的 CSS 属性,都是不继承父元素的。这时,可以强制它们继承。

* {
  font-family: inherit;
  line-height: inherit;
  color: inherit;
}

html {
  font-size: 125%;
  font-family: sans-serif;
  line-height: 1.5;
  color: #222;
}

上面代码中,首先强制所有元素的font-familyline-heightcolor属性都继承父元素,然后在网页的根元素html上设置具体的值。

em

em是一种相对单位,1em等于当前元素的font-size。一般来说,浏览器默认的字体大小是16像素,所以1em默认等于16像素。

h1 { font-size: 20px } /* 1em = 20px */
p { font-size: 16px } /* 1em = 16px */

上面代码中,对于h1元素,1em相当于20px;对于p元素,1em相当于16px

如果当前元素没有指定字体的像素大小,那么1em等于父元素字体的像素大小。这一点非常重要,务必牢记。

html { font-size: 16px }
h1 { font-size: 2em } /* 16px * 2 = 32px */

上面代码中,h1元素的字体大小是2em,但是它没有指定像素字体大小,因此2em等于2倍的父元素像素字体大小,也就是32px

em的一大好处是,很容易保持不同元素之间的比例关系,因此它比像素单位更合适用来设定字体大小。

html { font-size: 100%; }
h1 { font-size: 2em; }
p { font-size: 1em; }

上面代码中,h1的字体大小就是p的2倍。

整个网页的字体大小,如果全部使用em单位,不使用像素单位,会引发一个问题。em将会基于父元素计算,从而出现累积效应。

html { font-size: 100%; } /* 默认是 16px */
div { font-size: 2em; } /* 字体大小 32px */
p { font-size: 1em; } /* 字体大小 32px */

上面代码中,htmldiv的父元素,所以div1em是基于html计算的,因此2em字体大小等于32px。另一方面,divp的父元素,所以p1em实际上也是32px

除了字体大小,其他许多属性也可以使用em,比如marginpadding,相当于间接由font-size决定。

.header {
  font-size: 16px;
  padding: 0.5em 0.75em; /* 相当于 8px 12px */
  background: #7F7CFF;
}

上面代码中,.header元素的字体大小是16px,因此1em等于16px,所以padding就相当于8px 12px

这里比较混淆的是,如果font-size也使用em,两者的计算基点是不一样的。

h1 {
  font-size: 2em; /* 1em = 16px */
  padding: 1em;  /* 1em = 32px */
}

上面代码中,font-size是基于父元素计算的,如果父元素的字体大小是16px,那么font-size就是32pxpadding是基于font-size计算的,由于h1font-size32px,所以padding就是32px

由于以上的累积效应和计算基点差异的原因,造成em不容易精确控制,实际开发中往往改用rem

rem

rem单位与em几乎完全一致,只有一个差异,它总是等于根元素<html>font-size大小,与当前元素或父元素的设置无关,这就避免了em的缺陷。

h1 {
  font-size: 2rem; /* 1rem = 16px */
  margin-bottom: 1rem;
}

p {
  font-size: 1rem; /* 1rem = 16px */
  margin-bottom: 1rem;
}

上面代码中,不管<h1><p>在什么位置,1rem的大小总是不变的。

那么,何时使用rem,何时使用em呢?一个规则open in new window是字体大小font-size属性使用rem,其他必须等比例缩放的属性使用em。下面是一个例子。

button {
  font-size: 0.875rem;
  padding: .5em 1em;
  border: .125em solid #e3e3e3;
  @media (min-width: 48rem){ /* min-width: 768px */
    font-size: 1.125rem;
  }
  @media (min-width: 62rem){ /* min-width: 992px */
    font-size: 1.375rem;
  }
}

上面代码中,随着屏幕宽度的变化,字体大小会跟着改变,paddingborder会始终保持比例关系。

vh,vw

vh表示百分之一的浏览器视口高度,vw表示百分之一的浏览器视口宽度。每当视口的高度和宽度发生变化,它们就会自动重新计算。

html { font-size: 3vw; }

上面代码中,如果视口宽度增加,字体大小也会增加;视口宽度减小,字体大小也会减小。但是,如果宽度太小(比如320px),这样算出来的字体会太小(3px);如果宽度太大(比如1440px),字体又会太大(43px)。因此,可以考虑使用calc函数。

html { font-size: calc(18px + 0.25vw); }

vmin,vmax

vmin表示vhvw之中较短的那个单位,vmax则表示较长的那个单位。

一般来说,PC的屏幕是屏宽大于屏高,手机的屏幕是屏高大于屏宽。所以,很可能会出现,某一个区域在PC屏幕中宽度较小,在手机屏幕中宽度较大。这时,这两个单位就可以派上用处了。

h1 {
  font-size: 20vmin;
}

注意,上面的h1使用vmin单位时,当宽屏设备的视口宽度缩小时,它的字体大小是不变的,因为视口的高度没有改变。

ch

ch表示多少个字符。

width: 40ch;

上面代码表示宽度为40个字符。

calc()

calc方法用于计算值,常用于两种不同的单位之间的计算(比如百分比和绝对长度)。

实例1。每行放置4张图片,可以采用如下的代码。

img {
  float: left;
  width: calc(25% - 20px);
  margin: 10px;
}

实例2。动态排列图片,可以配合media query。


img {
  float: left;
  margin: 10px;
  width: calc(100% * 1 / 4 - 20px);
}

@media (max-width: 900px) {
  img {
    width: calc(100% * 1 / 3 - 20px);
  }
}

@media (max-width: 600px) {
  img {
    width: calc(100% * 1 / 2 - 20px);
  }
}

@media (max-width: 400px) {
  img {
    width: calc(100% - 20px);
  }
}

attr()

attr()用于读取网页元素的属性值。

div[data-line]:after { 
	content: "[line " attr(data-line) "]"; 
}

参考链接