Vue Scoped CSS深度解析:原理、误区与最佳实践

引言

在Vue开发中,Scoped CSS是一个强大而复杂的功能。它允许我们将样式限制在特定组件内,但同时也带来了一些细微的行为,可能导致意外的样式"泄漏"。本文将深入探讨Vue Scoped CSS的工作原理,解释常见的误区,并提供最佳实践建议。

Scoped CSS的基本原理

当我们在Vue组件中使用<style scoped>时,Vue会使用PostCSS对CSS进行转换。这个过程包括:

  1. 为组件的元素添加一个唯一的数据属性(例如data-v-f3f3eg9)。
  2. 重写CSS选择器,使其包含这个唯一属性。

例如,以下代码:

<template><div class="example">Hello</div>
</template><style scoped>
.example { color: red; }
</style>

会被转换为:

<div class="example" data-v-f3f3eg9>Hello</div>
.example[data-v-f3f3eg9] { color: red; }

这样,样式就被限制在了特定的组件内。

父子组件的Scoped CSS行为

当涉及到父子组件时,Scoped CSS的行为会变得复杂。让我们看一个例子:

<!-- ParentComponent.vue -->
<template><div class="parent"><ChildComponent /></div>
</template><style scoped>
.parent { color: blue; }
.child { color: red; }
</style><!-- ChildComponent.vue -->
<template><div class="child"><p>Child content</p></div>
</template><style scoped>
.child { color: green; }
</style>

在这个例子中:

  1. 父组件的.parent样式会正常应用,不会影响子组件。
  2. 父组件的.child样式会影响到子组件,尽管子组件有自己的scoped样式。

解密data-v属性继承

这里有一个常见的误解需要澄清。当我们说"子组件的根元素继承父组件的作用域"时,并不意味着子组件的根元素获得与父组件相同的data-v-属性。实际情况如下:

  1. 唯一标识符:每个带有scoped样式的组件都有自己唯一的data-v-标识符。
  2. 子组件根元素:子组件的根元素实际上同时接收到自己的data-v-属性和父组件的data-v-属性。
  3. 内部元素:子组件内部的元素只接收子组件自己的data-v-属性。

让我们看一个具体的例子:

<!-- ParentComponent.vue -->
<template><div class="parent"><ChildComponent /></div>
</template><!-- ChildComponent.vue -->
<template><div class="child"><p>Child content</p></div>
</template>

渲染后的HTML如下所示:

<div class="parent" data-v-parent123><div class="child" data-v-child456 data-v-parent123><p data-v-child456>Child content</p></div>
</div>

注意:

  • 父div有data-v-parent123
  • 子组件的根div同时有data-v-child456data-v-parent123
  • 子组件内的p元素只有data-v-child456

为什么会这样?

  1. 根元素继承:子组件的根元素继承父组件的作用域,这允许父组件有意地对子组件进行样式设置。
  2. 隐式深度选择器:Vue对scoped样式中的类选择器和属性选择器应用了隐式的深度选择器行为。这不同于显式使用/deep/>>>,但效果类似。
  3. 组件边界处理:Vue对组件之间的边界和组件内部的边界处理方式不同,这就是为什么样式可能会以看似意外的方式"泄漏"到子组件中。

这种机制的影响

  1. 样式泄漏:父组件的样式可能会影响子组件的根元素,这既可能是有用的,也可能造成问题。
  2. 特异性增加:子组件的根元素由于有两个data-v-属性,其特异性更高,这可能使得样式覆盖变得棘手。
  3. 调试:在检查元素时,看到多个data-v-属性可以帮助你理解样式来源。

如何控制这种行为

  1. 提高选择器特异性:在子组件中使用更具体的选择器。
<!-- ChildComponent.vue -->
<style scoped>
.child-component .child { color: green; }
</style>
  1. 使用CSS Modules:考虑使用CSS Modules而不是scoped CSS,以获得更严格的封装。
<style module>
.child { color: green; }
</style>
  1. 采用BEM命名约定:使用BEM等命名约定来减少冲突的可能性。
<template><div class="child-component__container"><p class="child-component__text">Child content</p></div>
</template><style scoped>
.child-component__container { /* styles */ }
.child-component__text { /* styles */ }
</style>
  1. 使用Vue 3的:deep()伪类:在Vue 3中,可以使用:deep()伪类来明确控制深度选择器的行为。
<style scoped>
.parent :deep(.child) {/* 这会影响.parent内的.child,即使在子组件中 */color: red;
}
</style>

最佳实践

  1. 明确意图:当从父组件样式化子组件时,要尽可能具体,以避免意外的样式泄漏。
  2. 使用组件类:为你的组件添加识别性的类,使选择更加有意图性。
<template><div class="my-component"><!-- 组件内容 --></div>
</template><style scoped>
.my-component { /* 样式 */ }
</style>
  1. 谨慎使用全局样式:尽量避免在scoped样式中使用全局选择器。如果必须使用,请确保你完全理解其影响。
<style scoped>
/* 避免这样做 */
* { margin: 0; }/* 如果必须,请使用更具体的选择器 */
.my-component * { margin: 0; }
</style>
  1. 利用CSS变量:使用CSS变量可以在保持样式封装的同时提供一定的灵活性。
<!-- ParentComponent.vue -->
<style>
:root {--main-color: blue;
}
</style><!-- ChildComponent.vue -->
<style scoped>
.child {color: var(--main-color);
}
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/461959.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

位运算算法及习题 ,丢弃的数字 , 两整数之和 ,只出现一次的数字II

文章目录 位运算基础1.基础位运算2. 给一个数n,确定他的二进制位中的第x为是0还是13.将一个数n的二进制表示的第x位修改为14.将一个数n的二进制表示的第x位修改为05.位图的思想6. 提取一个数n二进制表示中最右侧的17. 去掉一个数n二进制表示中最右侧的18. 异或运算的运算律 丢弃…

使用form表单的action提交并接收后端返回的消息

使用form表单的action提交表单是同步提交的方式&#xff0c;会跳转页面&#xff0c;所以无法获取后端返回来到消息。这样描述或许没有太大感觉&#xff0c;如果我要通过表单的方式上传文件&#xff0c;并接收后台返回来的响应数据&#xff1b;这样说是不是就感同深受了呢。 1.…

曹操出行借助 ApsaraMQ for Kafka Serverless 提升效率,成本节省超 20%

本文整理于 2024 年云栖大会主题演讲《云消息队列 ApsaraMQ Serverless 演进》&#xff0c;杭州优行科技有限公司消息中间件负责人王智洋分享 ApsaraMQ for Kafka Serverless 助力曹操出行实现成本优化和效率提升的实践经验。 曹操出行&#xff1a;科技驱动共享出行未来 曹操…

2024年10月文章一览

2024年10月编程人总共更新了21篇文章&#xff1a; 1.2024年9月文章一览 2.《Programming from the Ground Up》阅读笔记&#xff1a;p147-p180 3.《Programming from the Ground Up》阅读笔记&#xff1a;p181-p216 4.《Programming from the Ground Up》阅读笔记&#xff…

【果蔬识别】Python+卷积神经网络算法+深度学习+人工智能+机器学习+TensorFlow+计算机课设项目+算法模型

一、介绍 果蔬识别系统&#xff0c;本系统使用Python作为主要开发语言&#xff0c;通过收集了12种常见的水果和蔬菜&#xff08;‘土豆’, ‘圣女果’, ‘大白菜’, ‘大葱’, ‘梨’, ‘胡萝卜’, ‘芒果’, ‘苹果’, ‘西红柿’, ‘韭菜’, ‘香蕉’, ‘黄瓜’&#xff09;…

Partition架构

优质博文&#xff1a;IT-BLOG-CN Partition架构 【1】结构&#xff1a; Region至少3个Zone&#xff0c;Zone内至少两个Partition&#xff0c;Partition内至少1个K8S Member Cluster&#xff1b; 【2】故障域&#xff1a; 故障域及核心链路至少Zone内收敛&#xff0c;甚至Part…

xlrd.biffh.XLRDError: Excel xlsx file; not supported

文章目录 一、问题报错二、报错原因三、解决思路四、解决方法 一、问题报错 在处理Excel文件时&#xff0c;特别是当我们使用Python的xlrd库来读取.xlsx格式的文件&#xff0c;偶尔会遇到这样一个错误&#xff1a;“xlrd.biffh.XLRDError: Excel xlsx file; not supported”。…

Java XML一口气讲完!(p≧w≦q)

Java XML API Java XML教程 - Java XML API SAX API 下面是关键的SAX API的摘要: 类用法SAXParserFactory创建由系统属性javax.xml.parsers.SAXParserFactory确定的解析器的实例。SAXParserSAXParser接口定义了几个重载的parse()方法。SAXReaderSAXParser包装一个SAXReader…

CTF顶级工具与资源

《Web安全》http://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247484238&idx1&snca66551c31e37b8d726f151265fc9211&chksmc0e47a12f793f3049fefde6e9ebe9ec4e2c7626b8594511bd314783719c216bd9929962a71e6&scene21#wechat_redirect 《网安面试指南》h…

腾讯云视频文件上传云存储时自动将mp4格式转码成m3u8

针对问题&#xff1a; 弱网环境下或手机网络播放mp4格式视频卡顿。 存储环境&#xff1a;腾讯云对象存储。 处理流程&#xff1a; 1&#xff1a;登录腾讯云控制台&#xff0c;进入对象存储服务&#xff0c;找到对应的存储桶&#xff0c;点击进入。 在任务与工作流选项卡中找…

【AIGC】逆向拆解OpenAI官方提示词Prompt技巧:高效提升ChatGPT输出质量

博客主页&#xff1a; [小ᶻZ࿆] 本文专栏: AIGC | ChatGPT 文章目录 &#x1f4af;前言&#x1f4af;OpenAI官方提示词的介绍OpenAI官方提示词的结构与组成如何通过分析提示词找到其核心组件 &#x1f4af;OpenAI官方提示词分析案例一&#xff1a;制定教学计划案例二&…

Linux之nfs服务器和dns服务器

NFS服务器 NFS&#xff08;Network File System&#xff0c;网络文件系统)&#xff0c;NFS服务器可以让PC将网络中的NFS服务器共享的目录挂载到本地端的文件系统中&#xff0c;而在本地端的系统 中看来&#xff0c;那个远程主机的目录就好像是自己的一个磁盘分区一样。 注&am…

【机器学习】25. 聚类-DBSCAN(density base)

聚类-DBSCAN-density base 1. 介绍2. 实现案例计算 3. K-dist4. 变化密度5. 优缺点 1. 介绍 DBSCAN – Density-Based Spatial Clustering of Applications with Noise 与K-Means查找圆形簇相比&#xff0c;DBSCAN可以查找任意形状和复杂形状的簇&#xff0c;如S形、椭圆、半圆…

计组-层次化存储结构

这里主要看存储的整体结构&#xff0c;cache&#xff0c;内存 这里看存储结构是按什么样的层次来划分存储结构&#xff0c;速度由慢到快&#xff0c;容量由大到小&#xff0c;这是基于性价比的考虑&#xff0c;所以分为多级多层次&#xff0c;可以做到提高速度的同时没有增加多…

奇瑞不客气智驾 晚不晚?

文/孔文清 一直很好奇&#xff1a; 尹同跃董事长的金句“智驾不客气”&#xff0c;应该怎么翻译成英语&#xff1f; 谷俊丽的演讲PPT给了我答案&#xff1a; All in Ai Cars ——全力以赴、全情投入智能化汽车。 谷俊丽是奇瑞全球创新大会上最兴奋的人之一&#xff0c;有一种闭…

【万兴科技-注册_登录安全分析报告】

前言 由于网站注册入口容易被黑客攻击&#xff0c;存在如下安全问题&#xff1a; 暴力破解密码&#xff0c;造成用户信息泄露短信盗刷的安全问题&#xff0c;影响业务及导致用户投诉带来经济损失&#xff0c;尤其是后付费客户&#xff0c;风险巨大&#xff0c;造成亏损无底洞…

vue3中跨层传递provide、inject

前置说明 在 Vue 3 中&#xff0c;provide 和 inject 是一对用于跨组件树传递数据的 API。它们允许你在祖先组件中使用 provide 提供数据或服务&#xff0c;然后在后代组件中使用 inject 来获取这些数据或服务。这种方式特别适用于跨多个层级的组件传递数据&#xff0c;而不需要…

JAVA——网络编程

目录 1.概述 常见的网络架构 2.网络编程三要素 a.ip b.端口 c.协议 3.UDP协议 a.InetAddress类 1.概述 2.实例化对象 b.DatagramSocket类&#xff08;快递公司&#xff09; c.DatagramPacket类&#xff08;包裹&#xff09; d.单播、组播、广播 4.TCP协议 …

完全透彻了解一个asp.net core MVC项目模板1

当我们使用Visual Studio 2022去新建一个基于asp.net core Web项目的时候&#xff0c;一般有三种选择&#xff0c;一种是空项目&#xff0c;一种是基于MVC的项目、再有一种就是基于包含Razor Pages实例的web应用。如下图&#xff1a; 今天&#xff0c;我们打算选择基于MVC模…

在米尔电子MPSOC实现12G SDI视频采集H.265压缩SGMII万兆以太网推流

1. 引言 随着网络视频平台的发展&#xff0c;用户对于4K高清画质的需求日益增长。然而&#xff0c;许多用户发现&#xff0c;即使购买了视频平台的会员&#xff0c;观看4K内容时画质却不如预期&#xff0c;有时甚至还会出现模糊、卡顿的情况。这种现象背后涉及到视频编码、网络…