BFC解读

欢迎访问我的博客https://qqqww.com/,祝所有码农同胞们早日走上人生巅峰,迎娶白富美~~

1 前言

在 Web 页面的 CSS 渲染中,BFC (Block Fromatting Context)块级格式化上下文,即 BFC 区域是一块独立的块级渲染区域,并且拥有一套渲染规则,来约束盒子布局,且与外部区域无关,块盒与行盒(行盒由一行中所有的内联元素所组成)都会垂直的沿着其父元素的边框排列

2 创建BFC

W3C 中对于 BFC 是这样说的:

浮动元素和绝对定位元素,非块级盒子的块级容器(例如 inline-blocks, table-cells, 和 table-captions),以及overflow值不为“visiable”的块级盒子,都会为他们的内容创建新的BFC(块级格式上下文)

简单粗暴来讲就是,满足以下任意一个条件或者多个条件即形成 BFC

  1. float 值不是 none
  2. overflow 值不是 visible
  3. display 值是 inline-block、table-cell、table-caption、flex 或 inline-flex
  4. position 值是 absoulute或fixed

所以只要满足了上述条件一个或多个,即形成 BFC

<div class = "container">
    内容区域
</div>
.container {
    overflow: hidden;
}

3 BFC的约束规则

浏览器对于BFC这块区域的约束规则如下:

  1. 在 BFC 区域,其内部元素会在垂直方向,一个接一个的排列,第一个元素的左上角顶部和 BFC 区域左上角顶部重合
  2. 且每一个盒子的左外边缘(margin-left)会触碰到容器的左边缘(border-left),(对于从右到左的格式来说,则触碰到右边缘)
  3. 垂直方向上的距离由margin决定,相邻两个兄弟元素(例如上兄弟元素的margin-bottom和下兄弟元素的margin-top会发生重叠),但与方向无关,水平方向也可以发生重叠,一般很少
  4. 每个元素都不会超过它的包含块,即不会超过 BFC 区域,浮动元素也不例外,而position为absolute的元素可以超出他的包含块边界
  5. BFC 区域是一块隔离的区域,与外界无关
BFC

4 BFC的应用

BFC 在页面布局中有很多应用,有很多其实已经用过,但是原理是要慢慢去领悟,例如:

  1. 垂直方向防止 margin 重叠
  2. 浮动元素会尽量接近左上角
  3. 为父元素设置 overflow: hidden;来清除浮动

.......等

4.1 margin重叠

同一个 BFC 区域的两个相邻子元素的相邻的margin会发生重叠

4.1.1 margin嵌套重叠

<!DOCTYPE html>
<html>  
<head> 
  <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  <!--The viewport meta tag is used to improve the presentation and behavior of the samples 
    on iOS devices-->
  <meta name="viewport" content="initial-scale=1, maximum-scale=1,user-scalable=no"/>
  <title></title>

  <style> 
    html, body { height: 100%; width: 100%; margin: 0; padding: 0; }
    #map{
      padding:0;
    }
    .first{
      margin:20px;
      background:lightgreen;
      width:100px;
      height:100px;
    }
    ul{
      /*display:inline-block;*/
      margin:10px;
      background:lightblue;
    }
    li{
      margin:25px;
    }
  </style> 
  
  
</head> 

<body class="claro"> 
  <div class="first"></div>
  <ul>
    <li>1</li>
    <li>2</li>
    <li>3</li>
  </ul>
</body> 

</html>

这里给了 .first盒子和ullimargin值分别为20px10px25px

在浏览器的控制面板调试如下图

此时,垂直方向上,div.fristul两个盒子之间的距离是25px,由.firstulli三者中最大的margin决定

margin

解决 margin 嵌套重叠:

  1. 如上例子,将例子中的 注释部分display:inline-block;去掉再去浏览器看看结果,就可以解决margin嵌套重叠的问题了,margin此时的值是二者之和30px

  2. 原理是:ul形成了一个BFC区域,它和上面的div.first都是独立的BFC区域,互不影响。

ul BFC区域中的问题:

但是此时,去看看ul中的limargin仍然是存在问题的,依然发生了margin嵌套重叠,因为此时li是属于ul这个封闭的BFC区域的

4.2 BFC中的浮动

当给一个父元素的子元素设置浮动的时候,父元素往往会发生高度塌陷,即父元素高度变为0

原因:浮动元素会脱离文档流(像绝对定位的元素也会脱离文档流),所以此时如果没有高度的容器的子元素是浮动元素,则该容器是不会被撑开的

常见的解决办法是:给父元素加overflow: hidden;清除浮动

原理:创建BFC的元素,子浮动元素也会参与其高度计算,即不会产生高度塌陷问题。实际上只要让父元素生成BFC即可,并不只有这两种方式

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Document</title>
    <style type="text/css">
        .container {
          background-color: green;
          /*overflow: hidden;*/
          /*display: inline-block;*/
          /*display: table-cell;*/
          /*display: flex;*/
          /*display: table-caption;*/
          /*display: inline-flex;*/
          /*float: left;*/
          /*float: right;*/
          /*position: absolute;*/
          /*position: fixed;*/

        }
        .container div {
          float: left;
          background-color: lightgreen;
          margin: 10px;
        }
    </style>
</head>
<body>
    <div class="container">
        <div>Sibling</div>
        <div>Sibling</div>
    </div>
</body>
</html>

将上段代码复制到你的编译器中,打开浏览器,打开控制面板,我们看一下,两个字元素是浮动的,父元素div.container`是没有高度的,原因正如上面解释的一样

BFCmargin

解决办法:将我上面代码中的注释去掉任意一个,都可以解决这种父元素塌陷的问题

原理正如上面解释的一样,创建BFC的元素,子浮动元素也会参与其高度计算,即不会产生高度塌陷问题。实际上只要让父元素生成BFC即可,并不只有这种方式

此时再去打开浏览器看看,父元素div.container就有高度了

拓展:一般清除浮动最常用的方式,双伪元素清除浮动,若子元素需要浮动,则给其父元素加一个类clearfix使其形成BFC即可

.clearfix:before, .clearfix:after {
    content: '';
    display: block;
    clear: both;
}
.clearfix {
    overflow: hidden; // 形成 BFC
}

4.3 多栏布局

与浮动元素相邻的已生成BFC的元素不能与浮动元素相互覆盖。利用该特性可以作为多栏布局的一种实现方式

<!DOCTYPE html>
<html>  
<head> 
  <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  <!--The viewport meta tag is used to improve the presentation and behavior of the samples 
    on iOS devices-->
  <meta name="viewport" content="initial-scale=1, maximum-scale=1,user-scalable=no"/>
  <title></title>

  <style> 
    html, body { height: 100%; width: 100%; margin: 0; padding: 0; }
    .left {
      background: yellow;
      float: left;
      width: 180px;
    }
    .center {
      background: blue;
      overflow: hidden;    
    }
    .right {
      background: red;
      width: 180px;
      float: right;
    }
  </style> 
  
  
</head> 

<body class="claro"> 
  <div class="container">
    <div class="left">
      <pre>
        .left {
          background: yellow;
          float: left;
          width: 180px;
        }
      </pre>
    </div>
    <div class="right">
       <pre>
        .right {
          background: blue;
          width: 180px;
          float: right;
        }
      </pre>
    </div>
    <div class="center">
    <pre>
        .center {
          background: red;
          overflow: hidden;
          height: 116px;
        }
      </pre>
    </div>
  </div>

</html>

这种布局的特点在于左右两栏宽度固定,中间栏可以根据浏览器宽度自适应

多栏布局1
多栏布局2
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,725评论 1 92
  • relative:生成相对定位的元素,通过top,bottom,left,right的位置相对于其正常位置进行定位...
    zx9426阅读 928评论 0 2
  • 先前在学习CSS float时,有同学提到了BFC这个词,作为求知好问的好学生,哪里不懂查哪里,那么今天就来研究一...
    这名字真不对阅读 6,542评论 3 19
  • 一、在什么场景下会出现外边距合并?如何合并?如何不让相邻元素外边距合并?给个父子外边距合并的范例 在CSS当中,相...
    dengpan阅读 560评论 0 0
  • 3、必要条件关系(*) 只有...才...(只有后边是重点) 注:提出问题+分析问题==解决问题(*) 反面论证的...
    琐珥阅读 569评论 0 0