//zxx: 为了演示方便,避免无谓的干扰,本文所有的尺寸均指水平尺寸,对应CSS均是宽度相关,例如width/min-width/max-width。

以前的我对flex-basis不屑一顾,以为就是个浮于表面的CSS属性。

最近深入研究后才发现,自己那是朱砂当红土,珍珠当泥丸,钻石当玻璃,檀木当柴火——完全不识货!

在Flex布局中,一个Flex子项的宽度是由元素自身尺寸,flex-basis设置的基础尺寸,以及外部填充(flex-grow)或收缩(flex-shrink)规则3者共同决定的。

本文只要讨论前两者对尺寸的表现影响,如果您对第3者对尺寸影响有兴趣,则可以参考“CSS flex属性深入理解”这篇文章。

直接进入正题。

更新于2020-05-17

当初写这篇文章的时候是边研究边写的,现在再看看,描述和结论没问题,但是描述的逻辑比较混乱,实际上在Flex布局中,一个flex子项的最终尺寸是基础尺寸、弹性增长或收缩、最大最小尺寸限制共同作用的结果。

其中:

基础尺寸由CSS flex-basis属性,width等属性以及box-sizing盒模型共同决定;

弹性增长指的是flex-grow属性,弹性收缩指的是flex-shrink属性;

最大最小尺寸限制指的是min-width/max-width等CSS属性,以及min-content最小内容尺寸。

最终尺寸计算的优先级是:

最大最小尺寸限制 > 弹性增长或收缩 > 基础尺寸

大家记住上面的尺寸应用规则,然后再看下面的研究细节,逻辑上会清晰很多。

更新于2021-02-07

随着近期Firefox改变了渲染策略(不再和IE一致),和Chrome/Safari浏览器保持了一致,我又重新审视了对flex-basis的理解,发现上面的理解某些细节还是有些问题。

例如,width属性应该归属于最小尺寸,不属于基础尺寸。

相关细节我会专门开一篇文章进行介绍。

一、flex-basis与盒模型

flex-basis的尺寸是作用在content-box上的,这个和width属性是一样的。

例如:

.flex-basis {

padding: 1em;

border: 1em solid deepskyblue;

color: deepskyblue;

flex-basis: 100px;

}

.width {

padding: 1em;

border: 1em solid deepskyblue;

color: deepskyblue;

width: 100px;

}

在外部尺寸够大,内容尺寸够小的情况下,两者的表现是一样的。

如下图示意:

我们可以通过设置box-sizing属性改变元素的盒模型,例如。

.flex-basis {

padding: 1em;

border: 1em solid deepskyblue;

color: deepskyblue;

flex-basis: 100px;

box-sizing: border-box;

word-break: break-all;

}

此时就可以看到内容展示宽度变小了,外部尺寸表现为100px:

更新于2020-05-16

现在又跑了一遍demo页面,Chrome和Firefox浏览器都不是100px的宽度了,不是上面截图的效果,难道Chrome浏览器改了策略了?仔细一测试,发现原来是后面的那个的demo页面中flex-basic那两列没有设置word-break:break-all导致的。

但是,在IE11浏览器下,box-sizing:border-box是没有任何效果的。

这个细节知识点还是有点用的,在开发内部系统的PC项目上,例如阅文内部系统,兼容Chrome和Firefox是没问题的,但顶不住老板用IE浏览器,因此,需要兼容到window 7 Edge,也就是IE 11,这个时候,就要避免改变Flex子项的box-sizing属性值,当然,移动端项目可以不用想那么多。

二、深入理解flex-basis和width的关系

大家一定要搞清楚这样一个事实,在Flex布局中,子项设置width是没有直接效果的。

此时一定会有人反驳,我明明设置了width:100px就有效果啊!

对是有效果,但并不是width直接生效的,而是flex-basis的作用。

深入理解flex-basis:auto

flex-basis的默认值是auto,表示自动,也就是完全根据子列表项自身尺寸渲染。

自身尺寸渲染优先级如下:

min-width > || max-width > width > Content Size

同时,在Flex布局中,flex-basis优先级是比width高的(可以理解为覆盖)。

所以,flex-basis和width同时设置了具体的数值,则width属性值直接被打入冷宫,在样式表现上完全被忽略。

例如:

项目1

项目2

项目3

项目4

CSS如下:

by-zhangxinxu {

display: flex;

}

item-basis-width {

padding: 1em;

border: 1px solid deepskyblue;

color: deepskyblue;

box-sizing: border-box;

width: 200px;

flex-basis: 100px;

}

结果width:200px完全是一个盒饭角色,所谓盒饭角色,就是明明参与了这部电视剧,也吃了剧组的盒饭,但是观众完全就没有意识到他的存在。这里的width属性就是这种性质,代码中确实出现过,但是样式渲染却完全体现不出来。

如下图所示,宽度表现为100px。

回到主线这里,实际上,flex-basis值是auto的时候,width属性值也是被打入冷宫的,只是,这个时候,由于后宫放权,所以,冷宫也是有影响力的。什么意思呢?

flex-basis:auto的含义是,子项的基本尺寸根据其自身的尺寸决定。而这个自身尺寸与下面这几个方面有关:

box-sizing盒模型(这个已经介绍过了);

width/min-width/max-width等CSS属性设置;

content内容(min-content最小宽度);

所以下面两段Flex布局中的CSS就不难理解了:

item-width {

width: 100px;

}

最终尺寸 = 自身尺寸 mix basis尺寸。

此时,自身尺寸为100px,basis尺寸按照自身尺寸来算(因为此时属性值是auto),因此最终尺寸是100px。

item-basis {

flex-basis: 100px;

}

最终尺寸 = 自身尺寸 mix basis尺寸。

此时,自身尺寸为content内容最小宽度,basis尺寸100px,因此最终尺寸是:如果content内容最小宽度不超过100px,则最终尺寸是100px,否则就是内容最小宽度。

记住上面的解释,我们再看下面几个例子,就会豁然开朗了。

案例1:相同表现

一个元素最小宽度超过100像素场景并不多,因此大部分场景下,flex-basis和width表现是一致的。

例如下面这个对比案例:

1. flex-basis:100px

项目1

项目2

项目3

项目4

2. width:100px

项目1

项目2

项目3

项目4

一个设置width:100px,一个设置flex-basis:100px:

item-width {

width: 100px;

}

item-basis {

flex-basis: 100px;

}

默认都是100px的展示,例如我的PC电脑显示器下:

由于默认flex-shrink属性值是1,因此,容器宽度不足时候的弹性收缩效果也是一样的,如下GIF示意:

案例2:不同表现

什么时候width:100px和flex-basis:100px表现不一样呢?就是最小内容宽度较大的时候,例如出现连续英文单词demo_by_zhangxinxu。

还是上面那个demo页面,点击任意某个子项,会改变元素里面的文字内容为demo_by_zhangxinxu。

此时就可以看到两者的差异了,如下图所示:

因为此时content内容最小宽度超过了100px,于是flex-basis:100px按照了最小内容宽度显示了;但是width:100px把元素的尺寸限制得死死的,字符直接溢出容器之外。

要想两者表现一致,可以使用word-break:break-all使单词断开。

其实,本文一开始的“flex-basic与盒模型样式表现”这个demo就已经展现了flex-basis和width属性的区别了。例如第4项设置的flex-basis值是100px,同时box-sizing是border-box,但是最终表现出来的尺寸却不是100px,却是个更大的宽度值,如下图所示:

主要是因为这一项里面的内容,'basis:100px'会当做一个独立的单词,并作为此时flex-basis默认就有的最小宽度,这个宽度值等同于min-content的计算尺寸。

我们可以对比一下隔壁width:100px的效果,可以看到文字内容'width:100px'直接溢出到了容器之外。

更深入的案例:最小内容宽度、width和flex-basis同时满足

根据上面的分析(不考虑容器尺寸不足或溢出),我们可以得到结论:

width:100px + flex-basis:auto = 元素自身100px

content + flex-basis:100px = max(content, flex-basis) = 大于等于100px

再加上flex-basis优先级是比width高的结论,我们就可以得到:

content + width:100px + flex-basis:100px = content + flex-basis:100px = max(content, flex-basis) = 大于等于100px

事实究竟是不是这样呢?我们看一个实际的例子。

CSS和HTML代码分别如下,还是使用我最爱的深天空蓝:

by-zhangxinxu {

display: flex;

}

item-zxx {

padding: 1em;

border: 1em solid deepskyblue;

color: deepskyblue;

box-sizing: border-box;

}

item-zxx:nth-child(1) {

width: 100px;

}

item-zxx:nth-child(2) {

flex-basis:100px;

}

item-zxx:nth-child(3) {

width: 100px;

flex-basis: 100px;

}

我仅设置了width:100px,没有设置了flex-basis:100px。

我没有设置width:100px,但是设置了flex-basis:100px。

我设置了width:100px,同时设置了flex-basis:100px

如果flex-basis优先级是比width高,则第3个子项的width:100px应该是忽略的,也就是项目2和项目3的渲染表现是一样的。

我们把上面的代码在各个浏览器下跑一下,结果发现出现了尴尬的事情:

在Chrome浏览器下,项目3和项目1表现是一样的,也就是width:100px依然起到了一些作用。

但是,在Firefox浏览器以及IE Edge浏览器下都是符合预期的项目3和项目2表现一样。

下面问题来了,到底哪个正确哪个错误呢?

按照历史的经验,大概率Chrome浏览器会在N年之后的什么时候修改这个表现,和Firefox等浏览器一致。

至于谁对谁错啊,这个是没有定论的,规范中估计没有对这个细节进行特别详细的描述。

那给我们的启示是什么东西呢?那很简单,flex-basis数值属性值和width数值属性值不要同时使用,就不会遇到浏览器发现差异的问题。说的更极端一点,在Flex布局中,除非万不得已,否则没有使用width属性的理由,请使用flex-basis代替。

三、flex-basis还支持关键字属性值

以前以为flex-basis只支持数值,最近才发现,flex-basis还支持一大波关键字属性值,包括:

/* 根据flex子项的内容自动调整大小 */

flex-basis: content;

/* 内部尺寸关键字 */

flex-basis: fill;

flex-basis: max-content;

flex-basis: min-content;

flex-basis: fit-content;

但是我要提前先泼一波冷水,目前,在Chrome浏览器下,上面任何一个关键字属性值都不支持。这是很罕见的场景。

其中,flex-basis:content是Edge 12以及Firefox浏览器支持:

max-content/min-content关键字目前仅Firefox浏览器支持。

fill和fit-content还没有任何浏览器支持。

由于最重要的Chrome浏览器不支持,因此上面几个关键字案例就不demo演示了,直接文字描述下他们的作用吧。

content

尺寸根据内容决定,跟我自己测试,表现和max-content接近。

max-content

最大内容宽度。

min-content

最小内容宽度。

fill

不详

fit-content

不详

四、flex-basis与min-width/max-width

很多前端在Flex布局的时候会使用min-width或者max-width限制列表上的最大尺寸和最小尺寸。

实际上很多时候都是没有必要的,活用flex-basis属性和flex属性,可以实现类似的效果。

例如我希望一个列表中的每一项最小宽度是100像素,尽可能填满整个容器,请问该如何实现?

无需使用min-width属性,可以试试下面的CSS代码:

.container {

display: flex;

flex-wrap: wrap;

}

.item {

flex-basis: 100px;

flex-grow: 1;

}

实现效果如下GIF示意:

如果希望换行的列表也是等宽的,则需要复制N多空标签即可。

五、最后小结一下

本文内容最后的总结一下:

flex-basis默认作用在content box上,IE11会忽略box-sizing;

width只是flex-basis为auto时候间接生效,其余时候使用优先级更高的flex-basis属性值;

flex子项元素尺寸还与元素内容自身尺寸有关,即使设置了flex-basis属性值;

flex-basis数值属性值和width数值属性值不要同时使用;

flex-basis还支持很多关键字属性,只不过目前兼容性不太好;

flex-basis使用得当可以实现类似min-width/max-width的效果。

关于Flex子项最终的尺寸也总结一下:是盒模型,元素自身尺寸特性,以及flex属性共同作用的结果。

想要彻底搞清楚啊,需要掌握一个体系的知识点,不是一件简单的事情。

本来想好好解释一下,一想到茫茫多啰里八嗦的东西,就算了,盒模型可以参考《CSS世界》,“元素自身尺寸特性”本文有提及,可以参考下,而要想了解flex属性三剑客的含义和作用可以参考文章以及上一篇文章“CSS flex属性深入理解”,总之,要学习和了解的东西很多。

行文匆忙,表达可能有些乱,有一些错误也在所难免,欢迎指正!

文中很多理解都属于个人理解,不一定准确,如果有不同意见,非常欢迎交流。

最后,感谢您阅读到这里,如果觉得文章写得还不错,欢迎转发或分享哦~

本文为原创文章,欢迎分享,勿全文转载,如果实在喜欢,可收藏,永不过期,且会及时更新知识点及修正错误,阅读体验也更好。

本文地址:https://www.zhangxinxu.com/wordpress/?p=9154

(本篇完)

css flex 知乎,Oh My God,CSS flex-basis原来有这么多细节相关推荐

  1. css布局方式_手把手教你CSS Flex布局「真香」

    作者:EcbJS 转发链接:https://blog.csdn.net/EcbJS/article/details/106466757 前言   之前做项目,关于布局方面没怎么深入研究过,好多页面都是 ...

  2. HTML+CSS制作知乎登录页面

    HTML+CSS制作知乎登录页面 ​ 由于没有用上js相关的技术,所以无法做出相应的功能,,只能做出知乎登录页面的用密码登录的界面,不能切换到免密码登录.相关界面的图片如下: 网页上的原图: 我做的: ...

  3. flex布局,弹性盒子,css使用及理解

    flex布局 以下内容主要分为四个部分: 常见概念 flex容器相关属性 flex元素相关属性 flex布局的应用 flex布局的兼容性处理 flex布局的相关概念 弹性盒模型 弹性盒模型(Flexi ...

  4. css 动态rem_【面试题】CSS知识点整理(附答案)

    目录 伪类和伪元素 实现固定宽高比(width: height = 4: 3)的div,怎么设置 CSS选择器 CSS解析规则 flex: 1 完整写法 display: none和 visibili ...

  5. apache伪静态把css 排除掉_一文梳理CSS必会知识点

    本文主要梳理CSS必会知识点,会持续补充更新哦!长文预警!这可能是目前最长的一篇了? ? ? ? CSS引入 有哪些引入方式?通过link和@import引入有什么区别?(* ) CSS引入方式有4种 ...

  6. css flexbox模型_5分钟内学习CSS Flexbox-初学者教程

    css flexbox模型 快速介绍流行的布局模块 (A quick introduction to the popular layout module) In this post, you'll l ...

  7. CSS基础工作原理(一)——css规则与选择符器

    css规则 为文档添加样式的三种方法:行内样式.嵌入样式.链接样式(此处按优先级排序) <div style="display:block">123</div&g ...

  8. css颜色rgba代码对照表_改善 CSS 的 10 个最佳实践

       戳蓝字「前端技术优选」关注我们哦! CSS 看起来是一种非常直接且不易犯错的语言.只需要添加规则以对网站进行样式设置就可以了,对吗?对于只需要几个 CSS 文件的小型站点,可能是这种情况.但是在 ...

  9. 谈谈一些有趣的CSS题目(十五)-- 谈谈 CSS 关键字 initial、inherit 和 unset

    开本系列,谈谈一些有趣的 CSS 题目,题目类型天马行空,想到什么说什么,不仅为了拓宽一下解决问题的思路,更涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题 ...

  10. 谈谈一些有趣的CSS题目(十四)-- 纯 CSS 方式实现 CSS 动画的暂停与播放!

    开本系列,谈谈一些有趣的 CSS 题目,题目类型天马行空,想到什么说什么,不仅为了拓宽一下解决问题的思路,更涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题 ...

最新文章

  1. python-range用法
  2. java耗时任务有哪些,java后台耗时任务多线程返回结果
  3. c语言双引号和单引号的区别
  4. VMware虚拟机与宿主无法复制的解决办法
  5. vc 通过句柄修改窗口大小_VC应用(1)通过VC修改销售订单行项目的字段
  6. 蔚来与雷蛇联合推出NIO ES6限量版车型 售价46.78万元
  7. Python:PyCharm提示Local variable ‘x‘ value is not used
  8. squid内存监控脚本
  9. NTP服务器搭建教程
  10. linux内核 quota,Linux系统磁盘配额(quota)
  11. [数据可视化] 折线图(Line Chart)
  12. H266 ISP 帧内子划分
  13. 服务器的主要防护手段有哪些
  14. 如何更好使用markdown输出pdf
  15. SEO优化:6个方法提升网站排名
  16. 必看~与众不同、通俗易懂的lol版Java学习路线图
  17. Symantec Backup Exec 2010 安装报 bad ELF interpreter: No such file or directory
  18. jzoj 3461. 【NOIP2013模拟联考5】小麦亩产一千八(math)
  19. java 设计模式.pdf 免费下载
  20. 方舟(ARK)物品指令代码!

热门文章

  1. appium命令版安装
  2. 【华为交换机配置命令大全】
  3. Java 13 –深入了解JDK的新功能
  4. 人脸检测颜值软件_百度AI人脸识别颜值-百度AI人脸识别测颜值源码下载-西西软件下载...
  5. Word:文字怎么变成方框了(转)
  6. idea翻译软件TKK网络连接超时
  7. [模仿]html5游戏_FlyppyBird
  8. springboot整合mybatis (三) 一对多配置
  9. 阴阳师服务器维护3月14,阴阳师手游抢先体验服3月14日维护更新公告
  10. 【C语言练习】1.1弹跳小球