对Flex布局的总结与思考
阅读本文之前最好对flex布局有基本了解,可以通过“参考资料”中列举的资源来学习。
flex布局规范的设计目标
一维布局模型(one-dimensional layout model),元素项沿着水平或垂直方向来排列,就像一条沿着一个方向的“流”。
与之对应的,CSS Grid Layout是一个二维布局模型。两者互为补充。
- 空间分配(space distribution),(假设主轴是水平方向)元素项的最终宽度受到当前行剩余空间(或不足空间)的影响,就像是有弹性一样会膨胀和收缩。
- 强大的对齐支持(align and justify),align和justify本质上来说,是定义多余(空白)的空间要放在哪里。align(align-items, align-content)定义了交叉轴方向上的多余(空白)空间分布,而justify(justify-content)定义了主轴方向的多余(空白)空间要分布。
为了方便讨论,我们假设主轴是水平方向,当主轴是垂直方向的时候是同理的。
对align和justify的思考
主轴方向的多余空间
justify-content定义的是主轴方向的多余空间要如何分布。
主轴方向的多余空间的出现是因为容器宽度 > 元素项宽度之和。如图:
这个可交互实例来自MDN。
等一下,不是说 【主轴方向的多余空间会分配给元素项->使元素项膨胀->元素项占满主轴的空间】 的吗?为什么这里又有多余的空间来给justify-content分发呢?这是因为元素项不一定会膨胀(flex-frow的默认值为0,默认不膨胀),即使膨胀,膨胀后的宽度也会受到max-width的约束。因此有很多时候,主轴在元素项膨胀以后还是有多余空间的。
一个行内,交叉轴方向的多余空间
align-items定义的是一个行内,交叉轴方向的多余空间要如何分布。
一个行内,交叉轴方向的多余空间的出现是因为行的高度大于项的高度。由于各个项的高度不一致,比较高的项会将整行的高度撑开,对于那些比较矮的项,在它的垂直方向上就会出现多余空间。如下图:
这个可交互实例来自MDN。
关于高度撑开的讨论,见用css控制元素高度:自底向上和自顶向下的方法。
行与行之间,交叉轴方向的多余空间
align-content定义的是行与行之间,交叉轴方向的多余空间要如何分布。
在这里说的“行”,指的是一个flex容器内,由于flex-wrap: wrap造成的换行(下面会讨论到换行),而不是指【第一个flex容器是一行,第二个flex容器是第二行】!
行与行之间,交叉轴方向的多余空间的出现是因为容器高度 > 容器内各行的高度之和。
前面说过,一个行的高度是由这一行中最高的项撑开的。一个flex容器,默认的时候(height:auto),其高度也是被其内部的所有行的高度撑开的,在这个时候容器的高度恰好等于所有行的高度之和,不存在“行与行之间,交叉轴方向的多余空间”。
但是如果容器本身定义了height: 10000px呢?它的高度就固定了,不会受到其内部的行的影响。这时候,如果所有行的高度之和不足以填满容器高度,交叉轴方向就会出现多余空间。如图:
这个可交互实例来自MDN。
对空间分配的思考
flex是如何计算项的宽度的?
所有项先按照原始宽度在容器中排列。
原始宽度由flex-basis决定,由于flex-basis默认值是auto(意思是取content-box的宽度作为flex-basis),因此一般width就是原始宽度。如果没有定义width,则width由项的子元素撑开。
如果容器有多余的宽度,则将这些多余宽度分配给每个项(分配比例由flex-grow控制),使得项的宽度增加,得到每个项的flex宽度。最终宽度基于flex宽度,但还会受到min-width、max-width的限制。
如果所有项的原始宽度已经超过了容器元素的宽度,那么会先检查flex-wrap是否允许换行,如果允许换行,则换行以后再计算flex宽度;如果不允许换行,则将超出的宽度分配给每个项(分配比例由flex-shrink控制),使得项的宽度减小,得到每个子元素的flex宽度。最终宽度基于flex宽度,但还会受到min-width、max-width的限制。
总结来说就是,width决定原始宽度,flex-grow/flex-shrink决定分配比例,min-width、max-width限制最终宽度。
flex宽度计算的例子(可在浏览器中打开):
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<div class="container">
<div class="flex">
<div class="content1">
<div class="inner1"></div>
</div>
<div class="content2">
<div class="inner2"></div>
</div>
</div>
</div>
</body>
<style>
.flex {
display: flex;
width: 1000px;
/* 容器元素的剩余剩余宽度将被分配 */
}
.content1 {
flex-grow: 1;
height: 100px;
background-color: chocolate;
/* 没有定义width,则由它的子元素撑开 */
}
.content2 {
flex-grow: 1;
height: 100px;
background-color: aqua;
/* 没有定义width,则由它的子元素撑开 */
}
.inner1 {
height: 50px;
background-color: cornflowerblue;
/* 将父元素的宽度撑开为100px */
width: 100px;
}
.inner2 {
height: 50px;
background-color: greenyellow;
/* width设置为百分比时,无法撑开父元素,因此父元素的原始宽度为0 */
width: 100%;
}
</style>
</html>
计算过程:
content1原始宽度100px,content2原始宽度0px。剩余宽度为1000px-100px=900px。
由于content的flex-grow都相等,因此剩余宽度被平均分配,每个content分到450px。
content1最终宽度100px+450px=550px,content2最终宽度0px+450px=450px。
开发时布局的一般流程
根据UI设计,确定需要多少“行”来显示所有内容,然后确定每一“行”有哪些“项”。一个“项”本身也可以成为容器,包含一行或多行。
这里的“行”指的就是一个
flex-direction:row的容器了(与之前的讨论不同)。它可以设置flex-wrap:wrap,使得一个“行”容器在宽度不够容纳子元素的时候,在容器内部产生换行。- 对每一“行”,定义其样式。行是一个flex容器(display:flex)。并使用justify-content、align-items来定义元素在容器中的分布方式。通过margin-top来定义行之间的纵向距离。
- 对每一“项”,定义其样式。使用margin、padding来对元素位置进行微调。合理使用flex-grow、flex-shrink、width、min-width、max-width来调整元素的宽度。通过margin-left来定义元素之间的横向距离。如果这“项”本身也是一个容器(包含一行或多行),返回第2步。
参考资料
本文转载于猿2048:对Flex布局的总结与思考
对Flex布局的总结与思考的更多相关文章
- flex布局帮助你快速实现布局
flex布局可以帮我们快速布局一些区块,实现你想要的效果,不用再去float,position之类的.我们在布局网页的时候很多时候都是一些特殊布局,flex就能帮我快速去布局,不需要去定位. 任何一个 ...
- flex布局大全 2019
有句话叫做:存在即是合理. 最近很喜欢flex布局模式,不过还在摸索中,这里正一边在项目中使用和总结,也在学习一些大牛们总结的东西和布局思考. 鉴于自己很苦恼,到处去ha资料,真的,就没有一个系统的, ...
- flex布局大全
有句话叫做:存在即是合理. 最近很喜欢flex布局模式,不过还在摸索中,这里正一边在项目中使用和总结,也在学习一些大牛们总结的东西和布局思考. 鉴于自己很苦恼,到处去ha资料,真的,就没有一个系统的, ...
- Flex 布局教程:语法篇
作者: 阮一峰 网页布局(layout)是CSS的一个重点应用. 布局的传统解决方案,基于盒状模型,依赖 display属性 + position属性 + float属性.它对于那些特殊布局非常不方便 ...
- Flex 布局教程:实例篇
该教程整理自 阮一峰Flexible教程 今天介绍常见布局的Flex写法.你会看到,不管是什么布局,Flex往往都可以几行命令搞定. 我的主要参考资料是Landon Schropp的文章和Solved ...
- 在移动端中的flex布局
flex布局介绍: flex布局很灵活, 这种布局我们也可以称之为弹性布局, 弹性布局的主要优势就是元素的宽或者高会自动补全; flex布局实例: 比如有两个div,一个div的宽度为100px, ...
- css flex布局
关于flex布局的一些简单用法 效果(下图) 实现代码: <!--html--> <div class="wrap"> <div class=&quo ...
- FLEX布局的一些问题和解决方法
前言 露珠最近研究了一下flex的布局方式,发现项w3c推出的这套布局解决方案对于日益复杂的前端开发布局来说是确实是一利器,并且在不同的屏幕上实现了真正的响应式布局:不再单纯地依赖百分比和float的 ...
- CSS之flex布局
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name ...
随机推荐
- JZ-035-数组中的逆序对
数组中的逆序对 题目描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对1000000007取模的结果输出. 即 ...
- python的变量与基本数据类型
今日内容 python多版本共存 python的注释 python的变量与常量 变量的本质 变量的命名规范 python基本数据类型 内容详细 python多版本共存 先将两个版本的python解释器 ...
- 二级py--day6数据库设计基础
二级py-- 数据库设计基础 1.数据定义语言(DDL):该语言负责数据的模式定义与数据的物理存取构建 2.数据操纵语言(DML):该语言负责数据的操纵,包括查询及总删改等操作 3.数据控制语言(DC ...
- Linux-本地日志服务管理(rsyslog基础)
目录 系统环境 1.常见的两种日志管理服务 1.1 RSYSLOG系统日志服务 1.2 ELK 2.RSYSLOG日志服务的相关知识 2.1 RSYSLOG日志消息级别 2.2 RSYSLOG日志服务 ...
- 安装Win7与Ubuntu16.04双系统操作教程
安装主要分为以下几步: 一. 下载Ubuntu 16.04镜像软件: 二. 制作U盘启动盘使用ultraISO: 三. 安装Ubuntu系统: 四. 用EasyBCD 创建启动系统启动引导: (根据个 ...
- idea创建web项目以及配置Tomcat
废话不多说,直接上干活: 1.在project中现创建好module,也就是java web项目 2.把路径名写清楚就行了 3.创建在WEB-INF上右击创建classes和lib以存储class编译 ...
- LCT板子
粘板子: #include<cstdio> #include<cstring> #include<algorithm> using namespace std; c ...
- JDK,JRE,JVM的作用及关系
1.作用 JVM:Java虚拟机,保证Java语言跨平台 JRE:Java程序的运行环境 JDK:Java程序的开发环境 2.关系 JRE:JVM+类库 JDK:JRE+工具
- 如何进行Hibernate的性能优化?
大体上,对于HIBERNATE性能调优的主要考虑点如下: l 数据库设计调整 l HQL优化 l API的正确使用(如根据不同的业务类型选用不同的集合及查询API) l 主配置参数(日志,查询缓存,f ...
- Java中会存在内存泄漏吗,请简单描述?
为了搞清楚Java程序是否有内存泄露存在,我们首先了解一下什么是内存泄露:程序运行过程中会不断地分配内存空间:那些不再使用的内存空间应该即时回收它们,从而保证系统可以再次使用这些内存.如果存在无用的内 ...