绝对定位的参考系是最近已定位祖先而非直接父元素;父元素需设position: relative等才能成为定位上下文,且top/left起点为padding box左上角。
很多人写 .child { position: absolute; top: 10px; left: 20px; },本意是让子元素离父容器左上角 20px/10px,结果却贴到了浏览器左上角——根本原因不是代码写错了,而是父元素没成为“定位上下文”。position: static(默认值)不构成定位上下文,哪怕它是直接父级,absolute 子元素也会一路往上找,直到遇到 position: relative、absolute、fixed 或 sticky 的祖先,或者干脆落到视口(initial containing block)。
position: relative(哪怕不设 top/left)transform(如 transform: translateZ(0))、filter 或 will-change,也会悄悄创建定位上下文——调试时要检查这些属性,不想要就显式写 transform: none
当父元素设置了 padding,比如 .container { position: relative; padding: 20px; },那么 .child { position: absolute; top: 0; left: 0; } 的左上角,会精确对齐到容器内容区(即 padding box)的左上角,也就是距离容器 border 内侧 20px 的位置。这不是 bug,是 CSS 规范明确定义的。

left: -20px
absolute + top/left,得换思路(比如用 margin 或 transform)为什么推荐 parent: relative + child: absolute?因为 relative 不脱离文档流,父容器能正常撑开高度、参与布局;而 absolute 子元素完全脱离流,不会干扰内部其他子元素排版——比如右上角徽章、下拉菜单、模态框遮罩层,全靠这套组合实现干净隔离。
parent: absolute:父元素一绝对定位,自身高度就塌陷,后代无法基于它做可靠布局parent: static + child: absolute:子元素基准飘忽,响应式或嵌套组件中极易错位position: relative —— 它零副作用,只干一件事:提供定位锚点z-index 只在同一个堆叠上下文(stacking context)内有效。如果父元素自己创建了新的堆叠上下文(比如加了 opacity: 0.99、transform、filter 或显式 z-index),那它的子元素的 z-index 就只在这个小世界里比大小,再大也压不住外部同级元素。
z-index: 9999,还是被盖住 → 检查父级是否无意触发了堆叠上下文outline: 1px solid red 确认边界z-index 提到公共祖先上,而不是分散在深层子元素position: relative 这行看似多余的代码。
来电咨询