CSS 有两个最重要的基本属性,前端开发必须掌握:display
和 position
。
display
属性指定网页的布局。两个重要的布局,我已经介绍过了:弹性布局flex
和网格布局grid
。
本文介绍非常有用的position
属性。我希望通过10分钟的阅读,帮助大家轻松掌握网页定位,说清楚浏览器如何计算网页元素的位置,尤其是新引进的sticky
定位。
本文由国内最大的在线教育平台之一"腾讯课堂"赞助。他们现在启动了"腾讯课堂101计划",推广平台上的课程资源,有不少优质内容。希望提高前端技术水平的同学,可以留意一下本文结尾的免费课程信息。
一、position 属性的作用
position
属性用来指定一个元素在网页上的位置,一共有5种定位方式,即position
属性主要有五个值。
static
relative
fixed
absolute
sticky
下面就依次介绍这五个值。最后一个sticky
是2017年浏览器才支持的,本文将重点介绍。
二、static 属性值
static
是position
属性的默认值。如果省略position
属性,浏览器就认为该元素是static
定位。
这时,浏览器会按照源码的顺序,决定每个元素的位置,这称为"正常的页面流"(normal flow)。每个块级元素占据自己的区块(block),元素与元素之间不产生重叠,这个位置就是元素的默认位置。
注意,static
定位所导致的元素位置,是浏览器自主决定的,所以这时top
、bottom
、left
、right
这四个属性无效。
三、relative,absolute,fixed
relative
、absolute
、fixed
这三个属性值有一个共同点,都是相对于某个基点的定位,不同之处仅仅在于基点不同。所以,只要理解了它们的基点是什么,就很容易掌握这三个属性值。
这三种定位都不会对其他元素的位置产生影响,因此元素之间可能产生重叠。
3.1 relative 属性值
relative
表示,相对于默认位置(即static
时的位置)进行偏移,即定位基点是元素的默认位置。
它必须搭配top
、bottom
、left
、right
这四个属性一起使用,用来指定偏移的方向和距离。
div { position: relative; top: 20px; }
上面代码中,div
元素从默认位置向下偏移20px
(即距离顶部20px
)。
3.2 absolute 属性值
absolute
表示,相对于上级元素(一般是父元素)进行偏移,即定位基点是父元素。
它有一个重要的限制条件:定位基点(一般是父元素)不能是static
定位,否则定位基点就会变成整个网页的根元素html
。另外,absolute
定位也必须搭配top
、bottom
、left
、right
这四个属性一起使用。
/* HTML 代码如下 <div id="father"> <div id="son"></div> </div> */ #father { positon: relative; } #son { position: absolute; top: 20px; }
上面代码中,父元素是relative
定位,子元素是absolute
定位,所以子元素的定位基点是父元素,相对于父元素的顶部向下偏移20px
。如果父元素是static
定位,上例的子元素就是距离网页的顶部向下偏移20px
。
注意,absolute
定位的元素会被"正常页面流"忽略,即在"正常页面流"中,该元素所占空间为零,周边元素不受影响。
3.3 fixed 属性值
fixed
表示,相对于视口(viewport,浏览器窗口)进行偏移,即定位基点是浏览器窗口。这会导致元素的位置不随页面滚动而变化,好像固定在网页上一样。
它如果搭配top
、bottom
、left
、right
这四个属性一起使用,表示元素的初始位置是基于视口计算的,否则初始位置就是元素的默认位置。
div { position: fixed; top: 0; }
上面代码中,div
元素始终在视口顶部,不随网页滚动而变化。
四、sticky 属性值
sticky
跟前面四个属性值都不一样,它会产生动态效果,很像relative
和fixed
的结合:一些时候是relative
定位(定位基点是自身默认位置),另一些时候自动变成fixed
定位(定位基点是视口)。
因此,它能够形成"动态固定"的效果。比如,网页的搜索工具栏,初始加载时在自己的默认位置(relative
定位)。
页面向下滚动时,工具栏变成固定位置,始终停留在页面头部(fixed
定位)。
等到页面重新向上滚动回到原位,工具栏也会回到默认位置。
sticky
生效的前提是,必须搭配top
、bottom
、left
、right
这四个属性一起使用,不能省略,否则等同于relative
定位,不产生"动态固定"的效果。原因是这四个属性用来定义"偏移距离",浏览器把它当作sticky
的生效门槛。
它的具体规则是,当页面滚动,父元素开始脱离视口时(即部分不可见),只要与sticky
元素的距离达到生效门槛,relative
定位自动切换为fixed
定位;等到父元素完全脱离视口时(即完全不可见),fixed
定位自动切换回relative
定位。
请看下面的示例代码。(注意,除了已被淘汰的 IE 以外,其他浏览器目前都支持sticky
。但是,Safari 浏览器需要加上浏览器前缀-webkit-
。)
#toolbar { position: -webkit-sticky; /* safari 浏览器 */ position: sticky; /* 其他浏览器 */ top: 20px; }
上面代码中,页面向下滚动时,#toolbar
的父元素开始脱离视口,一旦视口的顶部与#toolbar
的距离小于20px
(门槛值),#toolbar
就自动变为fixed
定位,保持与视口顶部20px
的距离。页面继续向下滚动,父元素彻底离开视口(即整个父元素完全不可见),#toolbar
恢复成relative
定位。
五、 sticky 的应用
sticky
定位可以实现一些很有用的效果。除了上面提到"动态固定"效果,这里再介绍两个。
5.1 堆叠效果
堆叠效果(stacking)指的是页面滚动时,下方的元素覆盖上方的元素。下面是一个图片堆叠的例子,下方的图片会随着页面滚动,覆盖上方的图片(查看 demo)。
HTML 代码就是几张图片。
<div><img src="pic1.jpg"></div> <div><img src="pic2.jpg"></div> <div><img src="pic3.jpg"></div>
CSS 代码极其简单,只要两行。
div { position: sticky; top: 0; }
它的原理是页面向下滚动时,每张图片都会变成fixed
定位,导致后一张图片重叠在前一张图片上面。详细解释可以看这里。
5.2 表格的表头锁定
大型表格滚动的时候,表头始终固定,也可以用sticky
实现(查看 demo)。
CSS 代码也很简单。
th { position: sticky; top: 0; }
需要注意的是,sticky
必须设在<th>
元素上面,不能设在<thead>
和<tr>
元素,因为这两个元素没有relative
定位,也就无法产生sticky
效果。详细解释可以看这里。
(正文完)
免费前端全栈课程
初学者刚接触前端,往往会被一大堆技术名词、框架和工具,搞得眼花缭乱。
到底哪些技术是目前的主流技术栈,既能用于公司的开发实务,又能为自己的简历增添亮点?
下面就是一套目前主流的前端技术栈。
(1)Node.js:服务器端的 JavaScript 运行环境,不管哪种前端开发,都必不可少的底层环境。
(2)Webpack:语法转换工具,把 ES6/TypeScript/JSX 语法转成浏览器可以运行的代码。
(3)Koa2:一个非常流行、简洁强大的 Node.js 后端的 Web 开发框架。
(4)MongoDB:目前应用最广泛的非关系数据库之一,功能丰富,用法较简单。
(5)Vue 全家桶:
- Vue:前端基础框架
- Vuex:配套的前端状态管理库。
- Vue Router:官方的路由插件,构建单页面应用必不可少。
- Vue CLI:脚手架工具,帮你快速上手 Vue 开发,无需再花多余时间去实现项目架构。
- Vant:有赞前端团队开发的轻量级移动端 Vue 组件库,让你快速使用已经封装好的各种页面组件。
看到这个名单,你是不是感到有点头大,全部掌握它们需要多少时间啊?
现在,腾讯课堂就有一门这样的课程,内容包含了所有这些工具,教你怎么用它们从头完成一个全栈项目,亲手做出一个手机端的移动商城,是由 慕课网的精英讲师--谢成老师讲授 。
这个课程原价98元, 活动期间,只要1块钱哦! 微信扫描下面的二维码,就可以领取优惠券,享受1元听课的福利。
该课程的制作单位是青盟科技。它是《腾讯课堂101计划》重点推广的优质机构,已有7年 IT 行业教学经验,培养收费学员2000+,有超过72%的学员都进入到名企大厂。如果你想了解课程的详细内容,获取课程大纲,或者想接受系统的前端培训,可以登录腾讯课堂查看"青盟科技"。
(完)
DeathGhost 说:
哈哈,有前端知识了,这个应该能看懂了我。
2019年11月19日 09:44 | # | 引用
DeathGhost 说:
sticky 还是比较方便的。比fixed
2019年11月19日 09:47 | # | 引用
littlePuppy 说:
sticky学习了
2019年11月19日 09:47 | # | 引用
xu 说:
元素绝对定位absolute的情况下,即使父元素是静态定位static,它相对的基点也不一定是html,有可能是有定位过(static除外)的祖先元素。
2019年11月19日 09:54 | # | 引用
windy 说:
很好用的属性,但是要兼容ie11(怨念~)
2019年11月19日 09:55 | # | 引用
李昂 说:
清晰简洁明了,感谢阮老师
2019年11月19日 09:55 | # | 引用
bigma 说:
其实就是向上找,直到找到一个不是static定位的元素为止
2019年11月19日 09:57 | # | 引用
史大林 说:
用position:sticky来固定表单的顶部看起来很美好,但是浏览器的兼容性会导致各种各样的问题。用caniuse查下就知道。之前做过类似的一个需求,最好的办法还是用js来固定,当初用的table-fixed-header这个jq插件。
2019年11月19日 10:05 | # | 引用
关宇 说:
阮老师,absolute有点问题:
“父元素不能是static定位,否则定位基点就会变成整个网页的根元素html”
这么描述准确些:
“相对于距离最近的有定位属性(一般是relative)的上级元素(一般是父元素)进行偏移,即定位基点是上级元素”
2019年11月19日 10:30 | # | 引用
jarvis 说:
学到了sticky
2019年11月19日 10:42 | # | 引用
阮一峰 说:
@xu,@关宇:
谢谢指出。我稍微改了一下,说明 absolute 定位的基点不能是 static 定位。
2019年11月19日 10:56 | # | 引用
浪过扬帆 说:
前排先赞一个再看,文章还热乎着呢
2019年11月19日 11:53 | # | 引用
大脑袋比目鱼 说:
内容很精彩,可以再结合MDN https://developer.mozilla.org/zh-CN/docs/Web/CSS/position
看,理解更好。
2019年11月19日 12:17 | # | 引用
cl2 说:
sticky, 学习了
2019年11月19日 14:47 | # | 引用
方应杭的学生 说:
fixed 要小心 transform
2019年11月19日 15:24 | # | 引用
hewei 说:
IE11 居然不支持 sticky 。否则做个双吸顶效果舒服一步到位了
2019年11月19日 15:32 | # | 引用
JavenLeung 说:
关于“absolute 定位的基点不能是 static 定位”那里,现在改的这个版本还是不太清晰,容易理解错
2019年11月19日 16:55 | # | 引用
Arman 说:
Absolute相对定位的元素应该是第一个非static定位的祖先元素或根元素吧
2019年11月19日 18:25 | # | 引用
Mr. Ma 说:
Bootstrap4 中的导航条也用到了 sticky 定位,不过 IE 还是用不了。
2019年11月19日 19:30 | # | 引用
Chao 说:
阮老师的文章就是硬
2019年11月20日 02:52 | # | 引用
陶某人 说:
涨知识了,之前没用过,用来试试
2019年11月20日 09:27 | # | 引用
国际庄hello范儿 说:
简单易懂,解了长期的疑惑。。。。。
2019年11月20日 13:52 | # | 引用
心斩心 说:
absolute定位基于最近的非static祖先元素。
mdn的解释可以看这儿:https://developer.mozilla.org/zh-CN/docs/Web/CSS/position
2019年11月20日 17:10 | # | 引用
John Hoo 说:
demo里面的th, 只设置了sticky和top属性, 没有设置所有的定位属性诶?
2019年11月21日 09:52 | # | 引用
John Hoo 说:
刚刚理解错了,再读一遍就理解了
2019年11月21日 09:54 | # | 引用
liugezhou 说:
sticky大法好
2019年11月21日 16:42 | # | 引用
刘骏佳 说:
我发现表格的demo切换时会有轻微的抖动,这个问题是怎么导致的?该如何解决呢?
2019年11月21日 19:04 | # | 引用
Traceless 说:
toolbar的overflow属性必须设置为visible,否则sticky不生效
2019年11月23日 17:49 | # | 引用
Neil Ning 说:
当设置position = absolute时,如果其祖先元素都没有设置定位,并且也没有为当前元素设置任何left/top/right/bottom的值, 其效果跟position = relative是一样的,元素并不会相对于根元素定位而出现在浏览器的左上角。
2019年11月26日 11:40 | # | 引用
微wx笑 说:
一直以为"动态固定"效果只能基于js实现,原来sticky也可以,调试的时候看到这个不知道什么作用,终于了解了。
2019年11月26日 20:03 | # | 引用
melon 说:
文章结尾出放的文章对于webpack的介绍有误。
2019年11月27日 17:51 | # | 引用
save 说:
温故而知新
2019年11月28日 17:54 | # | 引用
gehaoyu 说:
再看无数遍
2019年11月29日 17:45 | # | 引用
BugLi 说:
可以提一下 transform 和 fixed
2019年11月30日 15:49 | # | 引用
而井 说:
建议将【sticky生效的前提是,必须搭配top、bottom、left、right这四个属性一起使用,不能省略,否则等同于relative定位,不产生"动态固定"的效果】
修改为【sticky生效的前提是,必须搭配top、bottom、left、right这四个属性其中的一个或多个一起使用,不能省略,否则等同于relative定位,不产生"动态固定"的效果】,
否则容易产生歧义,会有人理解成要用sticky来达到“动态固定”的效果,必须同时设置四个属性。
2019年12月 1日 12:22 | # | 引用
Fighting_xiaobai 说:
sticky会了会了,感谢阮大大不吝赐教!
2019年12月 4日 10:19 | # | 引用
康康康康康 说:
原文:页面继续向下滚动,父元素彻底离开视口(即整个父元素完全不可见),#toolbar恢复成relative定位。
这一段感觉有点问题,应该是:当页面回滚到阈值一下,#toolbar 才会恢复成 relative 定位吧。
2019年12月 4日 10:47 | # | 引用
康康康康康 说:
不好意思 是我理解错了,阮老师想表达的和我的并不是一个意思。。
2019年12月 4日 11:07 | # | 引用
mirror 说:
你好 阮讲师:
这一点有些疑惑
"sticky生效的前提是,必须搭配top、bottom、left、right这四个属性一起使用,不能省略"
--
我在测试使用sticky定位时,只是写了top一个属性,也能有效.这个是怎么回事呢.
2019年12月10日 11:51 | # | 引用
gz668 说:
absolute和fixed定位会脱离“正常页面流”,导致下面的块元素上移,也就是对周围元素位置产生了影响,而文中所说不会对周围元素产生影响,这里应该怎么理解呢?有点疑惑。
2019年12月12日 20:39 | # | 引用
Can 说:
"sticky生效的前提是,必须搭配top、bottom、left、right这四个属性一起使用,不能省略,否则等同于relative定位,不产生"动态固定"的效果。"
你的实例都不用搭配啊,不也生效吗??
2019年12月16日 17:41 | # | 引用
tomatoyyk 说:
补充一点,粘性定位的祖先元素不能有overflow hidden,不然会失效,之前踩过这个坑
2020年1月 2日 18:08 | # | 引用
king 说:
sticky跟前面四个属性值都不一样,它会产生动态效果,很像relative和fixed的结合;
阮老师,我觉得sticky应该更像是static和fixed的结合吧?因为relative定位,在设置了top属性的情况下也会偏移,而sticky在未达到阀值的时候,表现更像是static,除非sticky情况下的top等四个属性只是用来设置阀值的
2020年6月 2日 17:52 | # | 引用
andy 说:
哎,如果四个边可以用不同的定位方式就好了
2020年8月 5日 16:43 | # | 引用
大衍 说:
如果对 html 进行 absolute, 又是相对于谁定位呢?
2020年11月27日 22:54 | # | 引用
Devon 说:
sticky 对父元素的overflow属性有要求
2020年12月 1日 17:57 | # | 引用
雨一直下 说:
没错,应该是逐级向上查找非static定位的父级元素,如果都没有,则以网页根节点(即:html)作为基点。
2021年8月20日 13:51 | # | 引用
虫子的世界 说:
我的补充总结:具体规则是当页面滚动的时候,父元素开始脱离视口,只要sticky元素与视口的距离达到生效门槛(必须至少搭配top、bottom、left、right中的一个来产生距离),它的定位模式就会切换到fixed;等到sticky元素开始超出父元素的范围,它的定位模式就会切换回relative,随后随着父元素一起滚出视口。可以这么理解:sticky保证了元素在父元素从开始脱离视口到父元素的空间尚可容纳该元素期间始终能完整呈现在视口中。
演示Demo:
https://jsbin.com/hiyohavove/1/edit?html,css,output
2022年1月 8日 01:29 | # | 引用
sunflower 说:
sticky的固定效果生效后,如何给生效的div加上阴影?或其它样式效果?
2022年4月 7日 23:16 | # | 引用
不懒熊 说:
感觉文中“它有一个重要的限制条件:定位基点(一般是父元素)不能是static定位,否则定位基点就会变成整个网页的根元素html。另外,absolute定位也必须搭配top、bottom、left、right这四个属性一起使用。”
这句话描述不是很准确,因为当元素的 position 属性设置为 absolute 时,该元素会脱离"正常页面流",并相对于最近的非 static 定位的祖先元素进行定位。也就是说如果它的父元素的是static,但是它的父元素的父元素不是static定位,则absolute定位的元素,则是以父元素的父元素进行定位,并不是根元素html。
2023年10月28日 01:01 | # | 引用