




选第几个?先搞清“数谁”::nth-child(n) 数父元素下所有子元素的总序号,:nth-of-type(n) 数同标签类型中的第几个;样式失效多因混淆“按位置选”与“按类型顺序选”。
根本区别就一句话::nth-child(n) 数的是父元素下**所有子元素的总序号**,:nth-of-type(n) 数的是**同标签类型中第几个**。样式不生效,90% 是因为没想清楚自己到底想“按位置选”还是“按类型顺序选”。
p:nth-child(2):只认“第二个座位”,且必须坐的是 p —— 如果第二个子元素是 div 或注释节点,哪怕后面有十个 p,它也不匹配任何元素p:nth-of-type(2):不管前面插了多少 h3、span 或空格文本,只要它是父元素里出现的第二个 p,就稳稳命中 或其他标签时,:nth-child 的序号会悄悄偏移,而 :nth-of-type 完全免疫这类干扰图文列表、新闻卡片、表单字段组等真实项目里,HTML 很少是纯 li 或纯 p 堆叠。一旦出现 h3、div、span 与目标标签交错,:nth-child 就容易“失焦”。
标题
导语段落
发布时间正文第一段
立即学习“前端免费学习笔记(深入)”;
正文第二段
p”加底色?写 .post p:nth-child(4) 看似对,但一旦 .meta 改成 或加了注释,序号就崩了.post p:nth-of-type(2) —— 直接锁定“第二个段落”,与中间穿插什么完全无关n 对不对两者都支持 odd、even、3n+1 这类表达式,计算逻辑也一样(n 从 0 开始代入,结果 ≥ 1 才生效)。但关键在于:这个“n”是套在哪个集合上算的。
li:nth-child(odd):在
的全部子节点中,取第 1、3、5… 个,且该节点必须是 li
li:nth-of-type(odd):先提取所有 li,再从中取第 1、3、5… 个 —— 即使它们在 DOM 中分别位于第 2、6、9 位ul > li)时,两者效果相同;但只要混入其他节点,结果就会分道扬镳别靠猜。打开浏览器开发者工具,在控制台直接运行以下命令,能直观看到哪些元素被选中:
document.querySelectorAll('.news-list p:nth-child(2)') // 返回空 NodeList?说明第2个子元素不是 p
document.querySelectorAll('.news-list p:nth-of-type(2)') // 返回一个元素?说明确实是第2个 pnth_chind)也会
!important 测试是否是权重问题,而非选择器本身失效:nth-child 更脆弱;若需稳定定位类型,:nth-of-type 是更鲁棒的选择实际项目里最常被忽略的点:文本节点也算子元素。HTML 中两个标签之间的换行和缩进,在 DOM 中就是真实的 Text 节点,它会占据一个“孩子”位置——这正是 :nth-child 失效最隐蔽的原因。