高度塌陷问题引发的清除浮动的方法

1. 高度塌陷原因分析

看下面的代码,总的父元素parent包含三个浮动的子元素,容器的高度不能自动伸长以适应内容的高度,出现了高度坍塌问题。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>高度塌陷问题</title>
    <style>
    .parent{
        margin:20px;
        background-color: red;
        border: 2px solid black;
    }
    .left,.center,.right{
        float: left;
        width: 200px;
        height: 200px;
        background-color: yellow;
        border: 2px solid blue;
    }
    </style>
</head>
<body>
    <div class="parent">
        <div class="left">left</div>
        <div class="center">center</div>
        <div class="right">right</div>
    </div>    
</body>
</html>

(最上面的一条黑线就是parent元素了,可以看到它的高度为0)

原因就是浮动使子元素脱离文档流,父元素无法感知子元素的存在,而且父元素内部不存在其他处于文档流中的元素,也就表现为高度为0。

既然是由浮动引起的,就用clear属性清除浮动吧,其实这是错误的,clear属性规定的是元素哪一侧不允许有其他浮动元素,但是我们并不是想让父元素周围没有其他浮动元素,而是减少浮动带来的影响,也就是使浮动元素闭合

2. 闭合浮动解决高度塌陷

(1)添加额外标签

通过在浮动元素末尾添加一个空的标签,例如 <div style=”clear:both”></div>,其他标签br等亦可。

    <div class="parent">
        <div class="left">left</div>
        <div class="center">center</div>
        <div class="right">right</div>
        <!-- 1. 添加空标签 -->
        <div class="clear"></div>
    </div>   
    /*清除浮动*/
    .clear{
        clear:both;
    }

优点:通俗易懂,容易掌握。 
缺点:可以想象通过此方法,会添加多少无意义的空标签,有违结构与表现的分离,在后期维护中将是噩梦,这是坚决不能忍受的,所以你看了这篇文章之后还是建议不要用了吧。

(2)使用 br标签和其自身的 html属性

这个方法有些小众,br 有 clear=“all | left | right | none” 属性

    <div class="parent">
        <div class="left">left</div>
        <div class="center">center</div>
        <div class="right">right</div>
        <!-- 2. 使用 br标签和其自身的 html属性 -->
        <br clear="all" />
    </div> 

优点:比空标签方式语义稍强,代码量较少 
缺点:同样有违 结构与表现的分离,不推荐使用

(3)父元素设置 overflow:hidden

通过设置父元素overflow值设置为hidden;在IE6中还需要触发 hasLayout ,例如 zoom:1;

    .parent{
        margin:20px;
        background-color: red;
        border: 2px solid black;
        /*父元素overflow值设置为hidden*/
        overflow: hidden;
    }

优点:不存在结构和语义化问题,代码量极少 。
缺点:内容增多时候容易造成不会自动换行导致内容被隐藏掉,无法显示需要溢出的元素;04年POPO就发现overflow:hidden会导致中键失效,这是我作为一个多标签浏览控所不能接受的。所以还是不要使用了。

(4)父元素设置 overflow:auto

同样IE6需要触发hasLayout,演示和3差不多

    .parent{
        margin:20px;
        background-color: red;
        border: 2px solid black;
        /*父元素overflow值设置为auto*/
        overflow: auto;
    }

优点:不存在结构和语义化问题,代码量极少 。
缺点:多个嵌套后,firefox某些情况会造成内容全选;IE中 mouseover 造成宽度改变时会出现最外层模块有滚动条等,firefox早期版本会无故产生focus等,  所以不要使用。

(5)父元素也设置浮动

优点:不存在结构和语义化问题,代码量极少 
缺点:使得与父元素相邻的元素的布局会受到影响,不可能一直浮动到body,不推荐使用

(6)使用:after 伪元素(推荐使用

需要注意的是 :after是伪元素(Pseudo-Element),不是伪类(某些CSS手册里面称之为“伪对象”),很多闭合浮动大全之类的文章都称之为伪类,不过csser要严谨一点,这是一种态度。

注意由于IE6-7不支持:after,使用 zoom:1触发 hasLayout

    <div class="parent clearfix">
        <div class="left">left</div>
        <div class="center">center</div>
        <div class="right">right</div>
    </div>  
    /*伪元素清除浮动*/
    .clearfix:after{
        content: '';
        display:block; 
        height: 0;
        clear: both;
    }
    .clearfix{
        zoom: 1;  /* 兼容ie6,触发IE hasLayout */
    }
    .parent{
        margin:20px;
        background-color: red;
        border: 2px solid black;
    }
    .left,.center,.right{
        float: left;
        width: 200px;
        height: 200px;
        background-color: yellow;
        border: 2px solid blue;
    }

(7)使用 :before和 :after 双伪元素

    .clearfix:before,.clearfix:after{
        display: table;
        content: "";
    }
    .clearfix:after {
        clear: both;
    }
    .clearfix {
        zoom: 1;
    }

3. 小结

通过对比,我们不难发现,其实以上列举的方法,无非有两类:

其一,通过在浮动元素的末尾添加一个空元素,设置 clear:both属性,after伪元素其实也是通过 content 在元素的后面生成了内容为一个点的块级元素;

其二,通过设置父元素 overflow 或者display:table 属性来闭合浮动。

4. 什么是hasLayout

IE使用Layout概念来控制元素的尺寸和位置。如果一个元素有Layout,它就有自身的尺寸和位置;如果没有,它的尺寸和位置由最近的拥有布局的祖先元素控制。
在默认情况下,拥有Layout的元素包括:

<html>, <body>
<table>, <tr>, <th>, <td>
<img>
<hr>
<input>, <button>, <select>, <textarea>, <fieldset>, <legend>
<iframe>, <embed>, <object>, <applet>
<marquee>
(注意,<p><div>默认不拥有Layout。)

凡是具有以下CSS属性的元素,也会拥有布局:

position: absolute
float: left|right
display: inline-block
width: any value other than 'auto'
height: any value other than 'auto'
zoom: any value other than 'normal' (IE专用属性)
writing-mode: tb-rl(IE专用属性)
overflow: hidden|scroll|auto(只对IE 7及以上版本有效)
overflow-x|-y: hidden|scroll|auto(只对IE 7及以上版本有效)

hasLayout是IE特有的属性,不是CSS属性。可以用JavaScript函数hasLayout查看一个元素是否拥有Layout。如果有,这个函数就返回true;否则返回false。hasLayout是一个只读属性,所以无法使用JavaScript进行设置。

参考: 高度坍塌问题--BFC模式解析

posted @ 2017-08-27 20:47  gq_orange  阅读(564)  评论(0编辑  收藏  举报