该内容主要整理关于 CSS 的相关面试题,其他内容面试题请移步至 「最新最全的前端面试题集锦」 查看。
content
)、填充(padding
)、边界(margin
)、 边框(border
);content
部分把 border
和 padding
计算进去了;标准盒模型
盒子总宽度 = width + padding + border + margin;
盒子总高度 = height + padding + border + margin
width/height 只是内容(content)高度,不包含 padding 和 border 值
IE 怪异盒模型
盒子总宽度 = width + margin;
盒子总高度 = height + margin;
也就是,width/height 包含了 content、 padding 和 border 值
通过 box-sizing 来改变元素的盒模型
CSS 中的 box-sizing
属性定义了引擎应该如何计算一个元素的总宽度和总高度
box-sizing: content-box;
默认的标准(W3C)盒模型元素效果,元素的 width
/height
不包含 padding
,border
,与标准盒子模型表现一致box-sizing: border-box;
触发怪异(IE)盒模型元素的效果,元素的 width/height
包含 padding
,border
,与怪异盒子模型表现一致box-sizing: inherit;
继承父元素 box-sizing
属性的值小结
content
)、内填充(padding
)、 边框(border
)、外边距(margin
)DOCTYPE
,内容宽高会包含内填充和边框,称为怪异盒模型(IE盒模型)width + padding + border + margin
width + margin
box-sizing: border-box
属性,触发“怪异模式”解析计算宽高详细介绍请移步至:对BFC的深层理解
块级格式化上下文,是一个独立的渲染区域,让处于
BFC
内部的元素与外部的元素相互隔离,使内外元素的定位不会相互影响。
IE下为Layout
,可通过zoom:1
触发
触发条件
HTML
元素position: absolute/fixed
display
的值为 inline-block
、table
、flex
、inline-flex
、grid
、inline-grid
float
值为 left
、right
overflow
值不为 visible
,为 auto
、scroll
、hidden
规则
BFC
的两个相邻 Box
垂直排列BFC
的两个相邻 Box
的 margin
会发生重叠BFC
中子元素的 margin box
的左边, 与包含块 (BFC
) border box
的左边相接触 (子元素 absolute
除外)BFC
的区域不会与 float
的元素区域重叠BFC
的高度时,浮动子元素也参与计算应用
margin
重叠div
都位于同一个 BFC
区域之中)示例
元素提升为一个比较特殊的图层,在三维空间中 (z轴) 高出普通元素一等。
触发条件
html
)position
flex
transform
opacity
filter
will-change
webkit-overflow-scrolling
层叠等级:层叠上下文在z轴上的排序
详细介绍见【居中布局】
!important
> 内联样式 = 外联样式 > ID选择器 > 类选择器 = 伪类选择器 = 属性选择器 > 元素选择器 = 伪元素选择器 > 通配选择器 = 后代选择器 = 兄弟选择器
!import
会覆盖页面内任何位置定义的元素样式style
属性写在元素内的样式*
)同一级别:后写的会覆盖先写的
css选择器的解析原则:选择器定位DOM元素是从右往左的方向,这样可以尽早的过滤掉一些不必要的样式规则和元素
1. 在浮动元素后面添加 clear:both
的空 div
元素
<div class="container">
<div class="left">div>
<div class="right">div>
<div style="clear:both">div>
div>
2. 给父元素添加 overflow:hidden
或者 auto
样式,触发 BFC
<style>
.container{
width: 300px;
background-color: #aaa;
overflow: hidden;
zoom:1; /*兼容IE*/
}
style>
<div class="container">
<div class="left">div>
<div class="right">div>
div>
3. 使用伪元素,也是在元素末尾添加一个点并带有 clear: both
属性的元素实现的【推荐】
推荐:不会在页面新增div,文档结构更加清晰
<style>
.clearfix{
zoom: 1; /*兼容IE*/
}
.clearfix:after{
content: "";
height: 0;
clear: both;
display: block;
visibility: hidden;
}
style>
<div class="container clearfix">
<div class="left">div>
<div class="right">div>
div>
link
功能较多,可以定义 RSS
,定义 Rel
等作用,而 @import
只能用于加载 csslink
时,页面会同步加载所引的 css,而 @import
所引用的 css 会等到页面加载完才被加载@import
需要 IE5
以上才能使用link
可以使用 js 动态引入,@import
不行详细介绍见【CSS3的新特性】
常见的动画效果有很多,如平移、旋转、缩放等等,复杂动画则是多个简单动画的组合
css实现动画的方式,有如下几种:
transition
实现渐变动画transform
转变动画animation
实现自定义动画详细介绍见【CSS动画和过渡】
opacity:0
:本质上是将元素的透明度将为0,就看起来隐藏了,但是依然占据空间且可以交互visibility:hidden
: 与上一个方法类似的效果,占据空间,但是不可以交互了overflow:hidden
: 这个只隐藏元素溢出的部分,但是占据空间且不可交互display:none
: 这个是彻底隐藏了元素,元素从文档流中消失,既不占据空间也不交互,也不影响布局z-index:-9999
: 原理是将层级放到底部,这样就被覆盖了,看起来隐藏了transform: scale(0,0)
: 平面变换,将元素缩放为0,但是依然占据空间,但不可交互传统的项目开发中,我们只会用到px
、%
、em
这几个单位,它可以适用于大部分的项目开发,且拥有比较良好的兼容性
从CSS3开始,浏览器对计量单位的支持又提升到了另外一个境界,新增了rem
、vh
、vw
、vm
等一些新的计量单位
利用这些新的单位开发出比较良好的响应式页面,适应多种不同分辨率的终端,包括移动设备等
在css单位中,可以分为长度单位、绝对单位,如下表所指示
CSS单位 | |
---|---|
相对长度单位 | em 、ex、ch、rem 、vw 、vh 、vmin、vmax、% |
绝对长度单位 | cm、mm、in、px 、pt、pc |
px(像素)
px
,表示像素,所谓像素就是呈现在我们显示器上的一个个小点,每个像素点都是大小等同的,所以像素为计量单位被分在了绝对长度单位中
有些人会把px
认为是相对长度,原因在于在移动端中存在设备像素比,px
实际显示的大小是不确定的
这里之所以认为px
为绝对单位,在于px
的大小和元素的其他属性无关
em
em
是相对长度单位。相对于当前对象内文本的字体尺寸。如当前对行内文本的字体尺寸未被人为设置,则相对于浏览器的默认字体尺寸(1em = 16px
)
为了简化 font-size
的换算,我们需要在css中的 body
选择器中声明 font-size= 62.5%
,这就使 em
值变为 16px*62.5% = 10px
这样 12px = 1.2em
, 10px = 1em
, 也就是说只需要将你的原来的px
数值除以 10,然后换上 em
作为单位就行了
特点:
em
的值并不是固定的em
会继承父级元素的字体大小em
是相对长度单位。相对于当前对象内文本的字体尺寸。如当前对行内文本的字体尺寸未被人为设置,则相对于浏览器的默认字体尺寸rem(常用)
html
的文字大小,达到等比缩放的功能html
最终算出来的字体大小,不能小于12pxhtml
的font-size:20px
的时候,那么此时的1rem = 20px
rem
做盒子的宽度,viewport
缩放vh、vw
vw
,就是根据窗口的宽度,分成100等份,100vw
就表示满宽,50vw
就表示一半宽。(vw
始终是针对窗口的宽),同理,vh
则为窗口的高度
这里的窗口分成几种情况:
像vw
、vh
,比较容易混淆的一个单位是%
,不过百分比宽泛的讲是相对于父元素:
position: absolute;
的元素是相对于已定位的父元素position: fixed;
的元素是相对于 ViewPort
(可视窗口)小结
px
:绝对单位,页面按精确像素展示%
:相对于父元素的宽度比例em
:相对单位,基准点为父节点字体的大小,如果自身定义了 font-size
按自身来计算(浏览器默认字体是 16px
),整个页面内1em
不是一个固定的值rem
:相对单位,可理解为 root em
, 相对根节点 html
的字体大小来计算vh
、vw
:主要用于页面视口大小布局,在页面布局上更加方便简单
vw
:屏幕宽度的1%
vh
:屏幕高度的1%
vmin
:取vw
和vh
中较小的那个(如:10vh=100px 10vw=200px
则vmin=10vh=100px
)vmax
:取vw
和vh
中较大的那个(如:10vh=100px 10vw=200px
则vmax=10vw=200px
)
a
标签有四种状态:链接访问前、链接访问后、鼠标滑过、激活,分别对应四种伪类:link
、:visited
、:hover
、:active
;
当链接未访问过时:
a
链接时,满足 :link
和 :hover
两种状态,要改变 a
标签的颜色,就必须将 :hover
伪类在 :link
伪类后面声明;a
链接时,同时满足:link
、:hover
、:active
三种状态,要显示a
标签激活时的样式(:active
), 必须将 :active
声明放到 :link
和 :hover
之后。因此得出 LVHA
这个顺序。:link
换成 :visited
。这个顺序能不能变?可以,但也只有
:link
和:visited
可以交换位置,因为一个链接要么访问过要么没访问过,不可能同时满足,也就不存在覆盖的问题。
calc
函数是css3新增的功能,可以使用 calc()
计算 border
、margin
、pading
、font-size
和 width
等属性设置动态值
注意点:
width: calc(100% - 10px);
calc()
函数支持 "+", "-", "*", "/"
运算;calc()
的浏览器,整个属性值表达式将被忽略。不过我们可以对那些不支持 calc()
的浏览器,使用一个固定值作为回退。css 引入伪类和伪元素概念是为了格式化文档树以外的信息。也就是说,伪类和伪元素都是用来修饰不在文档树中的部分
伪类
伪类存在的意义是为了通过选择器找到那些不存在DOM树中的信息以及不能被常规CSS选择器获取到的信息
a
标签的:link
、:visited
等,这些信息不存在与DOM树结构中,只能通过CSS选择器来获取;:first-child
来获取到。伪元素
伪元素用于创建一些不在文档树中的元素,并为其添加样式。
比如说,我们可以通过:before
来在一个元素前增加一些文本,并为这些文本添加样式。虽然用户可以看到这些文本,但是这些文本实际上不在文档树中。常见的伪元素有:::before
,::after
,::first-line
,::first-letter
,::selection
、::placeholder
等
伪类与伪元素的区别在于:有没有创建一个文档树之外的元素
::after
和 :after
的区别?
在实际的开发工作中,我们会看到有人把伪元素写成 :after
,这实际是 CSS2
与 CSS3
新旧标准的规定不同而导致的。
CSS2 中的伪元素使用1个冒号,在 CSS3 中,为了区分伪类和伪元素,规定伪元素使用2个冒号。所以,对于 CSS2 标准的老伪元素,比如 :first-line
,:first-letter
,:before
,:after
,写一个冒号浏览器也能识别,但对于 CSS3 标准的新伪元素,比如 ::selection
,就必须写2个冒号了
CSS3新增伪类有那些?
p:first-of-type
选择属于其父元素的首个
元素的每个
元素。p:last-of-type
选择属于其父元素的最后
元素的每个
元素。p:only-of-type
选择属于其父元素唯一的
元素的每个
元素。p:only-child
选择属于其父元素的唯一子元素的每个
元素。p:nth-child(2)
选择属于其父元素的第二个子元素的每个
元素。:after
在元素之前添加内容,也可以用来做清除浮动。:before
在元素之后添加内容。:enabled
表示任何已启用的元素。:disabled
控制表单控件的禁用状态。:checked
单选框或复选框被选中。image
对象,将其 src
属性设置为 webp
格式的图片,然后在 onload
事件中获取图片的宽高,如果能够获取,则说明浏览器支持 webp
格式图片。如果不能获取或者触发了 onerror
函数,那么就说明浏览器不支持 webp
格式的图片canvas
判断方法。 可以动态的创建一个canvas
对象,通过canvas
的toDataURL
将设置为webp
格式,然后判断返回值中是否含有image/webp
字段,如果包含则说明支持WebP
,反之则不支持根据页面渲染流程可得知:
文字单行超出显示省略号
div {
width: 200px;
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
}
文字多行超出显示省略号
div {
width: 200px;
display: -webkit-box;
-webkit-box-orient: vertical;
-webkit-line-clamp: 3;
overflow: hidden;
}
该方法适用于WebKit浏览器及移动端。
跨浏览器兼容方案:
div {
position:relative;
line-height:1.4em;
/* 3 times the line-height to show 3 lines */
height:4.2em;
overflow:hidden;
}
div::after {
content:"...";
font-weight:bold;
position:absolute;
bottom:0;
right:0;
padding:0 20px 1px 45px;
}
body {
filter: grayscale(100%); /* 百分比或者 0~1 */
}
可继承的只有:
颜色
、文字
、字体间距
、行高对齐方式
、列表样式
。
visibility
和 cursor
。letter-spacing
word-spacing
white-space
line-height
color
font
font-family
font-size
font-style
font-variant
font-weight
text-decoration
text-transform
direction
text-indent
text-align
list-style
list-style-type
list-style-position
list-style-image
将窗体自上而下分成一行一行,并在每行中按从左至右的挨次排放元素。
inline-block
inline-block
,且给父元素设 text-align: center
inline-block
可以用一排 a {display: inline-block}
实现横向导航栏,无论是居左的导航栏还是居右的都适用float
来实现,不过 inline-block
会比它好一些【原因:浮动会脱离文档流,导致父元素高度塌陷】设置了 position: fixed
固定定位属性的元素会脱离文档流,达到“超然脱俗”的境界。
也就是说此时给这种元素设置 top, left, right, bottom
等属性是根据浏览器窗口定位的,与其上级元素的位置无关。
但是有一种情况例外:
position: fixed
属性的元素,它的祖先元素设置了 transform
属性则会导致固定定位属性失效。transform
设置的不是none
,都会影响到 position: fixed
,因为此时就会相对于祖先元素指定坐标,而不是浏览器窗口。注意,这个特性表现,目前只在Chrome浏览器/FireFox浏览器下有。IE浏览器,包括IE11, fixed
还是fixed
的表现。
一、分析
在HTML中,每个元素都可以理解成一个盒子,在浏览器解析过程中,会涉及到回流与重绘:
具体的浏览器解析渲染机制如下所示:
HTML
,生成 DOM 树
,解析 CSS
,生成 CSSOM 树
DOM树
和 CSSOM树
结合,生成渲染树(Render Tree
)Layout
(回流):根据生成的渲染树,进行回流(Layout
),得到节点的几何信息(位置,大小)Painting
(重绘):根据渲染树以及回流得到的几何信息,得到节点的绝对像素Display
:将像素发送给GPU
,展示在页面上
- 在页面初始渲染阶段,回流不可避免的触发,可以理解成页面一开始是空白的元素,后面添加了新的元素使页面布局发生改变
- 当我们对
DOM
的修改引发了DOM
几何尺寸的变化(比如修改元素的宽、高或隐藏元素等)时,浏览器需要重新计算元素的几何属性,然后再将计算的结果绘制出来- 当我们对
DOM
的修改导致了样式的变化(color
或background-color
),却并未影响其几何属性时,浏览器不需重新计算元素的几何属性、直接为该元素绘制新的样式,这里就仅仅触发了重绘
二、如何触发
要想减少回流和重绘的次数,首先要了解回流和重绘是如何触发的
回流触发时机
DOM
元素offsetTop
offsetLeft
offsetWidth
offsetHeight
scrollTop
scrollLeft
scrollWidth
scrollHeight
clientTop
clientLeft
clientWidth
clientHeight
getComputedStyle
方法,原理是一样的回流过程:由于
DOM
的结构发生了改变,所以需要从生成DOM
这一步开始,重新经过样式计算、生成布局树、建立图层树、再到生成绘制列表以及之后的显示器显示这整一个渲染过程走一遍,开销是非常大的
重绘触发时机
通过构造渲染树和重排(回流)阶段,我们知道了哪些节点是可见的,以及可见节点的样式和具体的几何信息(元素在视口内的位置和尺寸大小),接下来就可以将渲染树的每个节点都转换为屏幕上的实际像素,这个阶段就叫做重绘
触发回流一定会触发重绘
DOM
的修改导致了样式的变化,并且没有影响几何属性的时候,会导致重绘(repaint
)。重绘过程:由于没有导致
DOM
几何属性的变化,因此元素的位置信息不需要更新,所以当发生重绘的时候,会跳过生存布局树和建立图层树的阶段,直接到生成绘制列表,然后继续进行分块、生成位图等后面一系列操作。
浏览器优化机制
offsetTop
等方法都会返回最新的数据三、如何避免
style
,而是采用修改 class
的方式。position
属性为 absolute
或 fixed
的元素上。DOM
,比如读取某元素 offsetWidth
属性存到一个临时变量,再去使用,而不是频繁使用这个计算属性;又比如利用 document.createDocumentFragment()
来添加要被添加的节点,处理完之后再插入到实际 DOM
中display: none
,操作结束后再把它显示出来。因为在 display
属性为 none
的元素上进行的 DOM
操作不会引发回流和重绘resize
、scroll
等进行防抖/节流处理。CSS3
的 transform
、opacity
、filter
这些属性可以实现合成的效果,也就是 CPU
加速。 举例分析
// TODO
在合成的情况下,会直接跳过布局和绘制流程,直接进入非主线程处理的部分,即直接交给合成线程处理。交给它处理有两大好处:
will-change
是CSS3
新增的标准属性,它的作用很单纯,就是"增强页面渲染性能",当我们在通过某些行为触发页面进行大面积绘制的时候,浏览器往往是没有准备,只能被动的使用CUP去计算和重绘,由于事先没有准备,对于一些复杂的渲染可能会出现掉帧、卡顿等情况。而will-change
则是在真正的行为触发之前告诉浏览器可能要进行重绘了,相当于浏览器把CUP拉上了,能从容的面对接下来的变形。
常用的语法主要有:
whil-change: scroll-position;
即将开始滚动will-change: contents;
内容要动画或者变化了will-transform;
transform
相关的属性要变化了(常用)注意:
will-change
虽然可以开启加速,但是一定要适度使用hover
中,这样移出元素的时候就会自动 remove
掉 will-change
了will-change
,注意要及时remove
掉,方式就是 style.willChange = 'auto'
比较常用的:
em
:定义字体大小时以父级的字体大小为基准;定义长度单位时以当前字体大小为基准。【例父级font-size: 14px
,则子级 font-size: 1em;
为 font-size: 14px;
;若定义长度时,子级的字体大小如果为14px
,则子级 width: 2em;
为 width: 24px
】rem
:以根元素的字体大小为基准。【例如html
的font-size: 14px
,则子级1rem = 14px
】%
:以父级的宽度为基准。【例父级width: 200px
,则子级width: 50%; height:50%;
为 width:100px; height:100px;
】vw
和vh
:基于视口的宽度和高度(视口不包括浏览器的地址栏工具栏和状态栏)。【例如视口宽度为1000px
,则60vw = 600px;
】vmin
和vmax
:vmin
为当前vw
和vh
中较小的一个值;vmax
为较大的一个值。【例如视口宽度375px
,视口高度812px
,则100vmin = 375px;
,100vmax = 812px;
】不常用的:
ex
和ch
:ex
以字符"x"
的高度为基准;【例如1ex
表示和字符"x"
一样长】ch
以数字"0"
的宽度为基准;【例如2ch
表示和2个数字"0"
一样长】移动端布局总结:
rem
和 flex
,可以结合各自的优点,比如 flex
布局很灵活,但是字体的大小不好控制,我们可以使用 rem
和媒体查询控制字体的大小,媒体查询视口的大小,然后不同的上视口大小下设置设置 html
的 font-size
。