理解 CSS 布局和块级格式上下文

本文的目的是介绍一些概念来帮你增强 CSS 码力。如标题所示这篇文章主要是讲块级格式上下文BFCBlock Formatting Context。你可能没听过这个术语但只要你曾经使用 过CSS 布局你就能明白它。理解 BFC 是什么、它如何工作、如何创建一个 BFC 是非常有用的这些能帮你更好的理解 CSS 布局。

这篇文章里我会通过几个你会很熟悉的的示例解释 BFC。我还会告诉你一个新的 display 值当你理解了 BFC 后可能会很需要这个值。

目录

什么是 BFC

一个简单的浮动的示例就能明白 BFC 的行为在下面的示例中我们创建一个 box 元素该元素包裹一段文字和一个浮动的图片。 如果文字内容多的话文字将环绕着整个浮动图片box 的边框会把他们整个包裹起来。

1

2

3

4

5

6

 

<div class="outer">

      <div class="float">I am a floated element.</div>

       I am text inside the outer box.

</div>

 

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

 

.outer {

      border: 5px dotted rgb(214,129,137);

      border-radius: 5px;

      width: 450px;

      padding: 10px;

      margin-bottom: 40px;

}

 

.float {

      padding: 10px;

      border: 5px solid rgba(214,129,137,.4);

      border-radius: 5px;

      background-color: rgba(233,78,119,.4);

      color: #fff;

      float: left;  

      width: 200px;

      margin: 0 20px 0 0;

}

 


文字环绕着浮动元素

但如果把一些文字删除就没有足够的文字去环绕图片浮动元素了同时由于浮动元素脱离文档流box 元素的边框高度就会随文字的减少而降低。


没有足够的文字box 元素边框的高度就会低于浮动元素的高度

之所以会发生这种情况是由于当我们浮动一个元素后box 元素仍然保持原来的宽度是文字所占的空间缩短了以给浮动元素腾出位置这就是为什么背景和边框能够看起来包裹住了浮动元素。

我们通常会使用两种不同的方式来解决这个问题一种是使用 clear hack就是在 文字和图片的下方插入一个 div 并将它的 CSS clear 属性设值为 both。另外一种方法是使用 overflow 属性 把它设值成非默认值 visible的值。

1

2

3

4

5

 

.outer {

      overflow: auto;

}

 


使用 overflow: auto 后 box 就能包裹浮动元素了

overflow 之所以能够有效是因为当它的是是非 visible 时会创建一个 BFC而 BFC 的功能之一就是包裹浮动元素。

BFC 是布局中的迷你布局

你可以把 BFC 当做你页面中的一块小布局当一个元素被创建成 BFC 后它其中的所有元素都会被它包裹。正如我们所见当 box 元素变成 BFC 后它其中的浮动元素就再也没能突破它的底部。除此之外BFC 还有一些有用的功能。

BFC 可以阻挡外边距叠加margins collapsing

理解外边距叠加是另外一个被低估的 CSS 技巧。在接下来的示例里我创建了一个背景灰色的 div这个 div 含有两个段落div 元素的 margin-bottom 为 40px同时每个段落都有 20px 的 margin-top 与 margin-bottom。

1

2

3

4

5

6

7

8

9

10

11

12

13

 

.outer {

       background-color: #ccc;

      margin: 0 0 40px 0;

}

 

p {

      padding: 0;

      margin: 20px 0 20px 0;

      background-color: rgb(233,78,119);

      color: #fff;

}

 

由于 p 元素的边缘与 outer 元素的边缘之间没有任何东西所以 outer 与 p 的 margin 会叠加p 会与 outer 的顶部与底部齐平p 对外的 margin 似乎与 outer 的 margin 合并了使我们无法在段落的上下看到 outer 的灰色背景。


由于 margin collapse外边距叠加我们看到 outer 内部上下没有灰色背景

如果我们把 outer 元素变成 BFC它就可以包裹住 p 以及 p 的 margin外边距不会发生叠加outer 元素内部就会出现由 p 元素的 margin 顶出来的上下灰色背景。

1

2

3

4

5

6

7

 

.outer {

       background-color: #ccc;

      margin: 0 0 40px 0;

      overflow: auto;

}

 


建立 BFC 后外边距不再叠加

一旦 BFC 建立它就会阻止它内部的元素逃离突破它。

一个 BFC 会停止去环绕浮动元素

你可能很熟悉 BFC 的这个特性我们在有浮动元素的列类型布局中常用到。如果一个元素创建了 BFC它就不会去环绕或者说包装任何浮动元素。看下面这个示例

1

2

3

4

5

6

 

<div class="outer">

      <div class="float">I am a floated element.</div>

      <div class="text">I am text</div>

</div>

 

class 名为 float 的元素将会浮动在布局的左侧class 名为 text 的 div 元素将会在它后面并环绕它。


文字环绕着浮动元素

我们可以通过给 text 元素建立 BFC 来阻挡这种环绕行为。

1

2

3

4

5

 

.text {

      overflow: auto;

}

 


text 元素建立 BFC 后就不再环绕浮动元素了

该方法也是我们创建浮动布局的基本方式。还需注意的是浮动一个元素时也会给该元素创建 BFC也就是说此时 .float 与 .text 都是 BFC这也是无论右侧高度低于还是高于左侧两者都不会互相围绕的原因。

创建一个 BFC 的常用方式

除了使用 overflow 外 一些其他的 CSS 属性也可以创建 BFC比如上面我们所见浮动一个元素也可以为该元素创建 BFC浮动元素会包裹它内部的所有元素。还有以下几种方式可以创建 BFC

使用 position: absolute 或者 position fixed

使用 display: inline-blockdisplay: table-cell 或者 display: table-caption其中 table-cell 和 table-caption是表格相关 HTML 元素的对应默认 CSS 值所以当你创建表格每个表格单元都会自动创建 BFC。

另外当使用 multi-column layout 多列布局时使用 colum-span: all 也可以创建 BFC。Flex弹性 和 Grid网格 布局中的元素也会自动创建类似 BFC 的机制只是它们被称为 Flex Formatting Context弹性格式上下文和 Grid Formatting Context(网格格式上下文)。这反映了它们所参与的布局类型。一个 Block Formatting Context块级格式上下文表明他内部的元素参与了块级布局一个 弹性格式上下文意味着它内部的元素参与了弹性布局。在实践中这几种布局的结果是相似的浮动元素会被包裹、外边距不会叠加。

创建 BFC 的新方式

使用 overflow 或其他的方法创建 BFC 时会有两个问题。第一个是这些方法本身是有自身的设计目的的所以在使用它们创建 BFC 时会可能产生副作用。例如使用 overflow 创建 BFC 后在某些情况下你可能会看到出现一个滚动条或者元素内容被削减。这是由于 overflow 属性是设计被用来让你告诉浏览器如何定义元素的溢出状态的。浏览器执行了它最基本的定义。

另一个问题是即使在没有出现副作用的情况下使用 overflow 也可能会使另一个开发人员感到困惑。他们可能会各种猜想这里为啥要把 overflow 的值设为 auto 或 scroll原开发人员做这个意义何在原开发人员是想让这里出现滚动条吗

最安全的做法应该是创建一个 BFC 时不会有任何副作用它内部的元素都安安全全的呆在这个小布局里这种方法不会引起任何意想不到的问题也可以让开发者意图清晰。CSS 工作组也十分认同这种想法所以他们定制了一个新的属性值display: flow-root

你可以使用 display: flow-root 安全的创建 BFC 来解决本文中提到的各种问题包括包裹浮动元素、阻止外边距叠加、阻止环绕浮动元素。


caniuse 上 display: flow-root 各浏览器支持情况

浏览器对该属性值的支持目前还是有限的如果你觉得这个属性值很方便请投票去让 Edge 也支持它。不过无论如何你现在应该已经理解了什么是 BFC以及如何使用 overflow 或其他方法来包裹浮动以及知道了 BFC 可以阻止元素去环绕浮动元素如果你想使用弹性或网格布局可以在一些不支持他们的浏览器中使用 BFC 的这些特性做降级处理。

理解浏览器如何布置网页是非常基础的。 虽然有时看起来无关紧要但是这些小知识可以加快创建和调试 CSS 布局所需的时间。

本文转载自前端记录http://www.ferecord.com/understanding-css-layout-block-formatting-context.html

时间: 2024-09-22 10:17:39

理解 CSS 布局和块级格式上下文的相关文章

详解CSS布局设计块元素和内联元素

css|设计|详解 块元素(block element)一般是其他元素的容器元素,块元素一般都从新行开始,它可以容纳内联元素和其他块元素,常见块元素是段落标签'P."form"这个块元素比较特殊,它只能用来容纳其他块元素. 如果没有css的作用,块元素会顺序以每次另起一行的方式一直往下排.而有了css以后,我们可以改变这种html的默认布局模式,把块元素摆放到你想要的位置上去.而不是每次都愚蠢的另起一行.需要指出的是,table标签也是块元素的一种,table based layout

深入理解JavaScript中的块级作用域、私有变量与模块模式_基础知识

本文详细的介绍了JavaScript中的块级作用域.私有变量与模块模式,废话就不多说了,具体如下: 1.块级作用域(私有作用域),经常在全局作用域中被用在函数外部,从而限制向全局作用域中添加过多的变量和函数. (function(count){ for(var i=0;i<count;i++){ console.log(i);//=>0.1.2.3.4 } console.log(i);//=>5 })(5); (function(){ var now=new Date(); if(no

深入理解CSS盒子模型

深入理解CSS盒子模型 前言:前阵子在做一个项目时,在页面布局方面遇到了一点小问题,于是上stackoverflow上求助.ifaou在帮助我解决我问题的同时,还推荐我阅读一篇有关CSS盒子模型的文章<The CSS Box Model>,阅读之后受益匪浅,才知道自己对盒子模型知识还是如此欠缺.恰逢学期末,项目验收后暂时告一段落,有空闲的时间.于是想把这篇文章翻译出来,一方面再给自己一点挑战和锻炼,另一方面也给大家参考,让更多的人受益. 这篇文章适合初级web设计朋友,让你对盒子模型有更近一步

[HTML/CSS]盒子模型,块级元素和行内元素

目录 概述 盒子模型 块级元素 行内元素 可变元素 总结 概述 在div+css中,了解块级元素和行内元素还是非常有必要的,比如:对行内元素使用width属性就会失效.虽然自己不是做前端的,但是,在项目中,曾经也弄过从前端布局,也吃过这方面的亏.今天,群里有朋友问起这个,就趁着学习一下,也算是查漏补缺吧,虽然,谈不上精通,但是了解,还是很有必要的. 盒子模型 css盒子模型分为两种,一种是遵循w3c标准的标准盒子模型,另外一种就是IE盒子模型. 标准盒子模型 IE盒子模型 通过上面两张图可以看出

CSS文档流与块级元素(block)内联元素(inline)

将窗体自上而下分成一行行, 并在每行中按从左至右的顺序排放元素,即为文档流. 每个非浮动块级元素都独占一行, 浮动元素则按规定浮在行的一端. 若当前行容不下, 则另起新行再浮动. 内联元素也不会独占一行. 几乎所有元素(包括块级,内联和列表元素)均可生成子行, 用于摆放子元素. 有三种情况将使得元素脱离文档流而存在,分别是浮动,绝对定位, 固定定位.   基于文档流, 我们可以很容易理解以下的定位模式: 相对定位, 即相对于元素在文档流中位置进行偏移. 但保留原占位. 绝对定位, 即完全脱离文档

CSS网页制作教程:display属性行内元素和块级元素

文章简介:内联(行内)元素.块级元素区别. A:行内就是在一行内的元素,只能放在行内:块级元素,就是一个四方块,可以放在页面上任何地方. B:说白了,行内元素就好像一个单词:块级元素就好像一个段落,如果不另加定义的话,它将独立一行出现.C:一般的块级元素诸如段落<p>.标题<h1><h2>....列表,<ul><ol><li> .表格<table>.表单<form>.DIV<div>和BODY<

CSS常见内联元素和块级元素

[块元素(block element) ] * address - 地址 * blockquote - 块引用 * center - 举中对齐块 * dir - 目录列表 * div - 常用块级容易,也是css layout的主要标签 * dl - 定义列表 * fieldset - form控制组 * form - 交互表单 * h1 - 大标题 * h2 - 副标题 * h3 - 3级标题 * h4 - 4级标题 * h5 - 5级标题 * h6 - 6级标题 * hr - 水平分隔线 *

JavaScript的作用域和块级作用域概念理解

  作用域永远都是任何一门编程语言中的重中之重,因为它控制着变量与参数的可见性与生命周期.讲到这里,首先理解两个概念:块级作用域与函数作用域. 什么是块级作用域呢? 任何一对花括号({和})中的语句集都属于一个块,在这之中定义的所有变量在代码块外都是不可见的,我们称之为块级作用域. 函数作用域就好理解了(*^__^*) ,定义在函数中的参数和变量在函数外部是不可见的. 大多数类C语言都拥有块级作用域,JS却没有.请看下文demo: //C语言 #include void main() { int

css position, display, float 内联元素、块级元素

position属性:position属性指出一个元素的定位方法.有4种可能值:static, relative, absolute or fixed: static:默认值,元素按照在文档流中出现的顺序渲染 absolute:绝对定位,元素相对于它的第一个被定位的祖先元素(非static)来进行定位 fixed:元素相对于浏览器window进行定位 relative:元素相对与它的正常位置进行定位,因此"left:20"就表示在元素正常位置的基础上,左移20像素的距离. float属