前端教程:一分钟带你揭开 BFC 神秘的面纱

本文由 简悦 SimpRead 转码, 原文地址 zhuanlan.zhihu.com

  1. 引言

在前端的布局手段中,一直有这么一个知识点,很多前端开发者都知道有它的存在,但是很多人也仅仅是知道它的存在而已,对它的作用也只是将将说得出来,可是却没办法说得非常的清晰。这个知识点,就是 BFC。想要了解 BFC 的规则,前提必须是熟悉前端网页的多种布局手段,例如盒的显示模式 display,三种布局手段标准流(normal)、浮动流(float)、定位流(position)等。你只有熟练掌握了这些布局手段之后,才能很好的理解 BFC。今天这篇文章,来大家解析一下 BFC,希望对各位新老朋友有所帮助。

  1. 定义

BFC - Block Formatting Context 块级格式化上下文 BFC 的定义,在官方文档到中,是这么介绍 BFC 的。

A block formatting context contains everything inside of the element creating it that is not also inside a descendant element that creates a new block formatting context.

强行翻译一下吧,简单来说,这句话的意思就是:

一个 BFC 区域包含创建该上下文元素的所有子元素,但是不包括创建了新的 BFC 的子元素的内部元素

很显然,哪怕强行翻译了,大部分人依旧是看不懂这句话的。看都看不懂,那自然就没什么能把它说明白。_talk is cheap, show me the code._看不懂意思,我用代码来给你演示。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
<div>
     <div></div>
     <div></div>
     <div></div>
     <div>
         <div></div>
         <div></div>
         <div></div>
     </div>
</div>

用这段代码来解释上面那段 BFC 定义的话,就应该是这个意思:#HM_bfc1 是一块 BFC 区域,这块区域包含了 box2、box3、box4、box5,也就是所有 HM_bfc1 的子元素。同时

HM_bfc2 也创造了一块 BFC 区域,包含了 box6,box7,box8。注意,第一个 box1 的 BFC,只包括 box1 的子元素 box2345,不包括 box678。HM_bfc2 这个 BFC 同样也仅仅是包括自己的子元素 box678。

划重点

  1. 每一个 BFC 区域只包括其子元素,不包括其子元素的子元素。(这 1 点比较容易理解)
  2. 每一个 BFC 区域都是独立隔绝的, 互不影响! (这点不太好理解,但是后续会使用代码验证)

看完上面的描述,很多朋友依旧不懂,把第 2 节用心的再读一遍,相信你会有新的收获。然后往下继续阅读,你会豁然开朗。

  1. 触发 BFC

1
并不是任意一个元素都可以被当做BFC,只有当这个元素满足以下任意一个条件的时候,这个元素才会被当做一个BFC。

触发 BFC 的条件

  • body 根元素
  • 设置浮动,不包括 none
  • 设置定位,absoulte 或者 fixed
  • 行内块显示模式,inline-block
  • 设置 overflow,即 hidden,auto,scroll
  • 表格单元格,table-cell
  • 弹性布局,flex

上代码说明

https://pic3.zhimg.com/v2-2a18c269d73b852789e675d9b670a57a_b.jpg

首先, body 元素是 1 个 BFC,因为它满足我们的第 1 个条件 (body 根元素),这个 BFC 区域包含子元素 hm1234,但是不包括两个 p 标签,需要注意的是,hm3 不是一个 BFC 区域,因为他不满足上面任意 1 个条件。如果我们希望 hm3 也是 1 个 BFC 区域,只要让 hm3 满足上面任意一个条件即可。

https://pic2.zhimg.com/v2-6d98ba3229225332bc0b29c44d0bafdd_b.jpg

这个时候,hm3 元素被设置为了 overflow 为 hidden,满足上面第 5 个条件,所以此时,hm3 就成为了一个 BFC 区域,这个 BFC 区域包含其所有子元素 – 两个 p 标签。

划重点:

  1. 并不是所有的元素都是 BFC, 只有满足了上面的任意 1 个条件之后,这个元素才成为 1 个 BFC。
  2. 一个 BFC 区域,只包含其所有子元素,不包含子元素的子元素.
  3. 利用 BFC 解决问题

1
在你明白了解BFC的触发规则之后,那么就需要利用BFC的特点来解决我们在布局中遇到的一些问题了,还记得我们之前说过,BFC有一个特点是:**每一个BFC区域都是相互独立,互不影响的。**

4.1 解决外边距的塌陷问题(垂直塌陷)

​ 开发中,前端的布局手段,离不开外边距 margin,那么,也会遇到一些问题,例如外边距的垂直塌陷问题。

https://pic4.zhimg.com/v2-7cff5bfb8c5ab3ef8fcf9987995aaa47_b.jpg

通过以上的实例,我们会发现,代码给两个 div 盒子,都添加了四个方向的 margin,讲道理,学过数学的都知道,100+100=200. 可是,盒子之间的距离,现在却之后 100px。这就是很典型的 margin 的塌陷,两段 margin 重叠到了一块,互相影响。那么,如何利用 BFC,让这个问题得到解决呢。回忆下,上文说过,BFC,就是一个与世隔绝的独立区域,不会互相影响,那么,我们可以将这两个盒子,放到两个 BFC 区域中,即可解决这个问题。

https://pic1.zhimg.com/v2-c1b311e36d6ee8796e77e845eb2da05c_b.gif

4.2 利用 BFC 解决包含塌陷

当父子关系的盒子,给子元素添加 margin-top,有可能会把父元素一起带跑

https://pic3.zhimg.com/v2-c0fca1b00dc19fdb6b9832796c426bb6_b.jpg

原本,正确的显示方式,应该是粉色盒子与红色盒子的顶部距离为 50px,但是由于 margin 的塌陷问题,导致盒子内部的布局影响到了外部。这个时候,就可以触发 BFC,将父盒子变成一个独立的区域,这样在 BFC 区域内部的任何操作,都不会影响到外部。

https://pic3.zhimg.com/v2-e1bf321fff2d9cdc07b847fdb15321c6_b.gif

4.3 当浮动产生影响的时候,可以利用 BFC 来清除浮动的影响

https://pic3.zhimg.com/v2-1f35c422fb1094426b86528aca194a7a_b.jpg

以上代码表示,一个没有设置高度的父盒子,包含着七个子元素。如果此时,所有的子元素都浮动的话。

https://pic1.zhimg.com/v2-b4187b67a9e839201d0ef9edf3f44ef4_b.gif

当所有的子元素都浮动了,这个时候,父盒子失去了原有的高度,这就是浮动的影响。这个时候,同样也可用 BFC 的机制,来清除浮动带来的影响。使用 BFC,将所有的浮动元素包裹起来。

https://pic4.zhimg.com/v2-1d3938af99f8c503eb84260ecd823b03_b.gif

4.4 BFC 可以阻止标准流元素被浮动元素覆盖

https://pic2.zhimg.com/v2-f7b5596a5e1c99c03b77479b601a3235_b.jpg

以上情况,红色盒子浮动,蓝色盒子时标准流,默认情况下,浮动元素覆盖了标准流元素。但是,如果将蓝色盒子的 BFC 触发,那么情况将有所变化。

https://pic3.zhimg.com/v2-bad86c8d94e7cbe5b0bb38803f25604a_b.gif

当蓝色盒子触发了 BFC 之后,浮动元素再也不能覆盖它了,而且还能利用这个特性,来实现蓝色盒子宽度根据红色盒子的宽度来做自动适应

  1. 总结

  • 一个 BFC 区域只包含其子元素,不包括其子元素的子元素.
  • 并不是所有的元素都能成为一块 BFC 区域,只有当这个元素满足条件的时候才会成为一块 BFC 区域
  • 不同的 BFC 区域之间是相互独立的,互不影响的。利用这个特性我们可以让不同 BFC 区域之间的布局不产生影响.

好了,本期前端技术流文章就到这里了,期待下次的见面吧!