scoped原理、穿透原理、哈希计算

2024-01-30 05:36

本文主要是介绍scoped原理、穿透原理、哈希计算,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 什么是socped
  • 原理
  • data-v-xxxx

什么是socped

当一个style标签拥有scoped属性时,它的CSS样式就只能作用于当前组件,通过该属性,可以使组件之间的样式不相互污染。也就是实现组件私有化,起到样式隔离的作用

原理

  1. 组件实例生成一个唯一的标识,给组件中的每个标签对应的dom元素添加一个标签属性,data-v-xxxx,也叫做组件ID
  2. 给 < style scoped>中的每个选择器的最后一个选择器添加一个属性选择器,原选择器[data-v-xxxx],如:原选择器为.container #id div,则更改后的选择器为.container #id div[data-v-xxxx]

引出另外一个问题:
如果使用第三方组件,加了scoped之后就可能控制不到第三方组件中的样式了。(例如element-ui)

所以这时候需要样式穿透
样式穿透的写法有两种:/deep/::v-deep、>>>(这种写法比较老)

样式穿透原理:
scoped后选择器最后默认会加上当前组件的一个标识,比如[data-v-xxxx]用了样式穿透后,同样可以通过这个标签属性来对其进行样式权重的控制。不会在选择器后面追加[data-v-xxxx]

样式穿透原理指的是当元素存在嵌套时,某个元素可以直接使用父元素的样式,也就是说当我们给父元素定义一个样式时,子元素可以直接使用,不需要重新定义。这种机制叫做样式穿透原理。 原理解释:样式穿透原理是计算机中选择器的优先级选择原理,即当选择器存在嵌套时,父元素中定义的样式可以被子元素继承,同时子元素也可以重新定义样式,且优先级将高于父元素的样式

data-v-xxxx

xxxx值是怎么得到的?
webpack+vue-loader对vue2的处理

// vue-loader/src/index.tsconst shortFilePath = path.relative(rootContext || process.cwd(), filename).replace(/^(..[/\])+/, '').replace(/\/g, '/')const id = hash(isProduction? shortFilePath + '\n' + source.replace(/\r\n/g, '\n'): shortFilePath)

vite+@vitejs/plugin-vue对vue3的处理

// vite-plugin-vue/src/util/descriptorCache.tsimport path from "node:path";
import { createHash } from "node:crypto";
import slash from "slash";function getHash(text) {return createHash("sha256").update(text).digest("hex").substring(0, 8);
}// 获取文件相对路径
const normalizedPath = slash(path.normalize(path.relative(root, filename)));
// 计算 ID
descriptor.id = getHash(normalizedPath + (isProduction ? source : ""));

可以发现,不管是 vue-loader 还是 @vitejs/plugin-vue ,data 属性 ID 的生成机制都是一样的,即:

  • 开发环境下会根据文件相对路径生成唯一 ID,比如 vite 中 src/App.vue 固定生成 7a7a37b1
  • 生产环境下会根据文件相对路径+文件内容共同生成唯一 ID
    注意:相同路径结构的 Vue 子应用的组件,在开发环境下会产生样式冲突,但在生产环境下大概率不会,除非文件内容完全一样。
    那如果遇到了冲突问题,除了手动修改文件路径或文件名,还有什么办法可以完全避免?

给 Vue 提 PR ! 一个更好的 ID 计算方式是加上项目名(或者 package.json 的 name),并支持手动指定,这样就可以彻底避免冲突问题了。

import path from "node:path";
import { createHash } from "node:crypto";
import slash from "slash";function getHash(text) {return createHash("sha256").update(text).digest("hex").substring(0, 8);
}// 获取项目名
const projectName = config.projectName || path.basename(root)
// 获取文件相对路径(含项目名)
const normalizedPath = slash(path.normalize(path.join(projectName, path.relative(root, filename))));
// 计算 ID
descriptor.id = getHash(normalizedPath + (isProduction ? source : ""));

为什么开发环境和生产环境的 ID 计算方式不一样?

首先,开发环境下最好不要加入文件内容进行 hash 计算。
这很好理解:

一来 hash 计算是耗时的,内容越多耗时越长;
二来还会频繁变动节点样式,徒增成本。

那生产环境为什么还要加入文件内容计算 hash ?
如果 ID 与文件内容无关,就可以实现稳定的 data 属性。对于 E2E (端对端)测试用例,就可以直接使用 data 属性进行元素寻址。

为什么在生产环境中要将文件内容纳入哈希计算?

在生产环境中,将文件内容纳入哈希计算主要是为了解决缓存问题和版本控制
在Web开发中,为了提高网站的加载速度和性能,通常会将静态资源(如CSS、JavaScript文件)进行缓存。当浏览器首次加载网页时,会将这些资源下载到本地,并根据资源的URL进行缓存。当用户再次访问同一网页时,浏览器会先检查缓存,如果资源没有发生变化,则直接使用缓存的资源,从而提高加载速度。

然而,如果在生产环境中对静态资源进行修改,而URL没有发生变化,浏览器可能仍然使用缓存的旧资源,导致网页显示不正确或出现错误。为了解决这个问题,可以使用哈希计算

哈希计算会将文件的内容计算出唯一的哈希值,并将其添加到文件名或URL中。当文件内容发生变化时,哈希值也会随之改变。这样,浏览器在加载网页时就会发现URL发生了变化,从而强制重新下载最新的静态资源。这保证了用户总是能够看到最新的文件版本,同时也解决了缓存问题。

这篇关于scoped原理、穿透原理、哈希计算的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

电脑系统Hosts文件原理和应用分享

《电脑系统Hosts文件原理和应用分享》Hosts是一个没有扩展名的系统文件,当用户在浏览器中输入一个需要登录的网址时,系统会首先自动从Hosts文件中寻找对应的IP地址,一旦找到,系统会立即打开对应... Hosts是一个没有扩展名的系统文件,可以用记事本等工具打开,其作用就是将一些常用的网址域名与其对应

Dubbo之SPI机制的实现原理和优势分析

《Dubbo之SPI机制的实现原理和优势分析》:本文主要介绍Dubbo之SPI机制的实现原理和优势,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Dubbo中SPI机制的实现原理和优势JDK 中的 SPI 机制解析Dubbo 中的 SPI 机制解析总结Dubbo中

Java如何将文件内容转换为MD5哈希值

《Java如何将文件内容转换为MD5哈希值》:本文主要介绍Java如何将文件内容转换为MD5哈希值的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java文件内容转换为MD5哈希值一个完整的Java示例代码代码解释注意事项总结Java文件内容转换为MD5

Java计算经纬度距离的示例代码

《Java计算经纬度距离的示例代码》在Java中计算两个经纬度之间的距离,可以使用多种方法(代码示例均返回米为单位),文中整理了常用的5种方法,感兴趣的小伙伴可以了解一下... 目录1. Haversine公式(中等精度,推荐通用场景)2. 球面余弦定理(简单但精度较低)3. Vincenty公式(高精度,

Android与iOS设备MAC地址生成原理及Java实现详解

《Android与iOS设备MAC地址生成原理及Java实现详解》在无线网络通信中,MAC(MediaAccessControl)地址是设备的唯一网络标识符,本文主要介绍了Android与iOS设备M... 目录引言1. MAC地址基础1.1 MAC地址的组成1.2 MAC地址的分类2. android与I

Spring框架中@Lazy延迟加载原理和使用详解

《Spring框架中@Lazy延迟加载原理和使用详解》:本文主要介绍Spring框架中@Lazy延迟加载原理和使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、@Lazy延迟加载原理1.延迟加载原理1.1 @Lazy三种配置方法1.2 @Component

spring IOC的理解之原理和实现过程

《springIOC的理解之原理和实现过程》:本文主要介绍springIOC的理解之原理和实现过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、IoC 核心概念二、核心原理1. 容器架构2. 核心组件3. 工作流程三、关键实现机制1. Bean生命周期2.

Redis实现分布式锁全解析之从原理到实践过程

《Redis实现分布式锁全解析之从原理到实践过程》:本文主要介绍Redis实现分布式锁全解析之从原理到实践过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、背景介绍二、解决方案(一)使用 SETNX 命令(二)设置锁的过期时间(三)解决锁的误删问题(四)Re

windows和Linux使用命令行计算文件的MD5值

《windows和Linux使用命令行计算文件的MD5值》在Windows和Linux系统中,您可以使用命令行(终端或命令提示符)来计算文件的MD5值,文章介绍了在Windows和Linux/macO... 目录在Windows上:在linux或MACOS上:总结在Windows上:可以使用certuti

redis中使用lua脚本的原理与基本使用详解

《redis中使用lua脚本的原理与基本使用详解》在Redis中使用Lua脚本可以实现原子性操作、减少网络开销以及提高执行效率,下面小编就来和大家详细介绍一下在redis中使用lua脚本的原理... 目录Redis 执行 Lua 脚本的原理基本使用方法使用EVAL命令执行 Lua 脚本使用EVALSHA命令