webpack plugin源码解析(四) HashedModuleIdsPlugin

news2024/11/27 0:18:17

文章目录

  • 作用
  • 涉及 webpack API
    • 获取chunkGraph
    • 获取当前编译过程中被使用过的 module id:compilation.usedModuleIds
    • 获取当前编译过程中所有的模块对象:compilation.modules
    • 判断 module 是否需要生成 id:module.needId
    • 获取指定module 的 module id:chunkGraph.getModuleId
    • 获取 module 属于多少个chunk:chunkGraph.getNumberOfModuleChunks
    • 获取 module 标识符:module.identifier
    • 设置 module id:chunkGraph.setModuleId
  • 实现
    • constructor
    • apply
    • getUsedModuleIdsAndModules

作用

  • 将模块打包生成后的 id 变成 hash 值,用于生成稳定的模块 id
new webpack.ids.HashedModuleIdsPlugin(),

在这里插入图片描述

涉及 webpack API

  • 获取chunkGraph

const chunkGraph = compilation.chunkGraph;
  • 获取当前编译过程中被使用过的 module id:compilation.usedModuleIds

compilation.usedModuleIds;  //其中一些模块可能会被排除在最终的构建结果之外,因为它们没有被使用过
  • 获取当前编译过程中所有的模块对象:compilation.modules

    • 包括那些被使用过的模块和那些没有被使用过的模块。
for (const module of compilation.modules) {
	// ...
}
  • 判断 module 是否需要生成 id:module.needId

    • 表示该模块是否需要一个模块 id, 在 webpack 的编译过程中,有些模块可能不需要一个独立的模块 id,例如一些内置模块或者一些被动态加载的模块
for (const module of compilation.modules) {
	if (!module.needId) continue;
	const moduleId = chunkGraph.getModuleId(module);
}
  • 获取指定module 的 module id:chunkGraph.getModuleId

for (const module of compilation.modules) {
	if (!module.needId) continue;
	const moduleId = chunkGraph.getModuleId(module);
}
  • 获取 module 属于多少个chunk:chunkGraph.getNumberOfModuleChunks

// chunkGraph.getNumberOfModuleChunks 表示 module 属于多少个chunk ,为 0 表示模块对象不属于任何一个 chunk
if (chunkGraph.getNumberOfModuleChunks(module) !== 0){
 // ...
}
  • 获取 module 标识符:module.identifier

    • 根据 module 的 type、request、layer创建
for (const module of compilation.modules) {
	module.identifier()
}

// module.identifier
identifier() {
	if (this.layer === null) {
		if (this.type === "javascript/auto") {
			return this.request; // "/xxx/Desktop/webpack/wb/node_modules/babel-loader/lib/index.js??ruleSet[1].rules[1].use[0]!/xxx/Desktop/webpack/wb/src/o1.js"
		} else {
			return `${this.type}|${this.request}`;
		}
	} else {
		return `${this.type}|${this.request}|${this.layer}`;
	}
}
  • 设置 module id:chunkGraph.setModuleId

for (const module of compilation.modules) {
	const moduleId= 'xxx'
	chunkGraph.setModuleId(module, moduleId);
}

实现

constructor

class HashedModuleIdsPlugin {
	/**
	 * @param {HashedModuleIdsPluginOptions=} options options object
	 */
	constructor(options = {}) {
		validate(options);

		/** @type {HashedModuleIdsPluginOptions} */
		this.options = {
			context: null,
			hashFunction: "md4",
			hashDigest: "base64",
			hashDigestLength: 4,
			...options
		};
	}
}

apply

apply(compiler) {
	const options = this.options;
	compiler.hooks.compilation.tap("HashedModuleIdsPlugin", compilation => {
		// hooks.moduleIds 为每个模块分配 id 时触发
		compilation.hooks.moduleIds.tap("HashedModuleIdsPlugin", () => {
			const chunkGraph = compilation.chunkGraph;
			const context = this.options.context // "/xxx/Desktop/webpack/wb"
				? this.options.context
				: compiler.context;
			// 获取所有效的 modules 和已经编译使用过的 module id
			const [usedIds, modules] = getUsedModuleIdsAndModules(compilation);
			// 对 modules 排序
			const modulesInNaturalOrder = modules.sort(
				compareModulesByPreOrderIndexOrIdentifier(compilation.moduleGraph)
			);
			for (const module of modulesInNaturalOrder) {
				// 获取 module 标识,用来创建 hash
				const ident = getFullModuleName(module, context, compiler.root); // "./node_modules/babel-loader/lib/index.js??ruleSet[1].rules[1].use[0]!./src/o1.js"
				
				const hash = createHash(options.hashFunction);
				hash.update(ident || "");
				const hashId = /** @type {string} */ (
					hash.digest(options.hashDigest)
				);
				
				let len = options.hashDigestLength;
				// 默认取 hash 前四位,如果有重复往后取
				while (usedIds.has(hashId.slice(0, len))) len++;
				const moduleId = hashId.slice(0, len);
				// 更改 module id
				chunkGraph.setModuleId(module, moduleId);
				// 添加新 module id
				usedIds.add(moduleId);
			}
		});
	});
}

getUsedModuleIdsAndModules

  • 获取所有效的 modules 和已经编译使用过的 module id
  • 只有需要生成 module id 的 module 且之前没有创建过且 module 在某个 chunk 中才有效
const getUsedModuleIdsAndModules = (compilation, filter) => {

	const chunkGraph = compilation.chunkGraph;

	const modules = [];

	const usedIds = new Set();
	// 用于存储在当前编译过程中被使用过的 module id。
	if (compilation.usedModuleIds) {
		for (const id of compilation.usedModuleIds) {
			usedIds.add(id + "");
		}
	}
	// compilation.modules: 存储在当前编译过程中所有的模块对象,包括那些被使用过的模块和那些没有被使用过的模块。
	for (const module of compilation.modules) {
		// module.needId 表示该模块是否需要一个模块 id, 在 webpack 的编译过程中,有些模块可能不需要一个独立的模块 id,例如一些内置模块或者一些被动态加载的模块
		if (!module.needId) continue;
		// 获取 module id 
		const moduleId = chunkGraph.getModuleId(module);
		
		if (moduleId !== null) {
			// 如果已有则跳过
			usedIds.add(moduleId + "");
		} else {
			// chunkGraph.getNumberOfModuleChunks 表示 module 属于多少个chunk ,为 0 表示模块对象不属于任何一个 chunk
			if (
				(!filter || filter(module)) &&
				chunkGraph.getNumberOfModuleChunks(module) !== 0
			) {
				// 将该 module 放入 modules 用于后续计算 module id hash 值
				modules.push(module);
			}
		}
	}

	return [usedIds, modules];
};

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

新品首发丨计讯物联智慧灯杆TG473-A6-5“滴水不进”,严苛环境下稳定运行无压力

近年来,智慧灯杆产业快速发展,已广泛应用于智慧城市、智慧交通、智慧园区、智慧停车等细分场景,提质增效、节本降耗的作用突显。值得注意的是,不同的场景有着不同的环境条件,如高低温、潮湿、沙尘等,对智慧…

Nginx学习(9)—— 负载均衡模块

文章目录Nginx负载均衡模块负载均衡配置指令钩子初始化配置初始化请求peer.get和peer.free回调函数小结Nginx负载均衡模块 负载均衡模块用于从”upstream”指令定义的后端主机列表中选取一台主机。nginx先使用负载均衡模块找到一台主机,再使用upstream模块实现与这…

应急响应 - Windows进程分析,Windows网络分析,tasklist,wmic process

「作者简介」:CSDN top100、阿里云博客专家、华为云享专家、网络安全领域优质创作者 「推荐专栏」:对网络安全感兴趣的小伙伴可以关注专栏《网络安全入门到精通》 进程网络排查一、Tasklist1、tasklist /v1、tasklist /svc2、tasklist /m二、wmic proces…

JAVAWeb02-CSS

1. CSS CSS 指的是层叠样式表 (Cascading Style Sheets) 1.1 概述 1.1.1 官方文档 地址: https://www.w3school.com.cn/css/index.asp 1.1.2 为什么需要 CSS 在没有 CSS 之前,我们想要修改 HTML 元素的样式需要为每个 HTML 元素单独定义样式属性,费心…

nodegui搭建/你好/打包

0、github连接问题 警告:如果你的网络有任何有任何有任何有任何有任何有任何有任何有任何有任何有任何连接 github 的问题,彻底放弃该框架 请转到其他框架 electron-egg教程、electron-egg官网,或其他electron项目 Tauri教程、Tauri官网 NW.…

Smartbi电子表格软件架构与差异化特色

选择和Excel结合Smartbi电子表格软件选择与Excel结合的原因在于,Excel一直被模仿,从未被超越。虽然市场上的报表软件很多,但存在太多的不完美。国外的产品功能复杂、难于学习(控件方式),做不了中国式复杂格…

SpringBoot使用Redis实现分布式缓存

✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。 🍎个人主页:Java Fans的博客 🍊个人信条:不迁怒,不贰过。小知识,大智慧。 💞当前专栏…

Matlab群体智能优化算法之巨型睡莲优化算法(VAO)

Matlab群体智能优化算法之巨型睡莲优化算法(VAO) 摘要:介绍一种新型智能优化算法,巨型睡莲优化算法。其应用于24个基准测试函数,并与其他10个著名算法进行了比较。提出的算法在10个优化问题上进行了测试:最小生成树、枢纽位置分配…

技术复盘(4)--docker

技术复盘--docker资料地址环境初始docker安装dockerdocker常用命令-都需要先启动dockerdocker容器打包为镜像docker坑docker卸载docker数据卷dockerFile构建过程搭建私有仓库--非图形化界面搭建私有仓库--图形化界面docker理论知识资料地址 docker官网:https://www…

一种供水系统物联网监测系统

1.1供水系统 1.1.1监测范围选择依据 (1)管网老化区域管网 管网建设年代久远,通常管网发生破损问题较大,根据管网本身属性和历史发生事件的统计分析,结合数理统计,优先选择管网老化区域的管段所在区域进行…

基于imx8m plus开发板全体系开发教程4:Linux系统开发

前言: i.MX8M Plus 开发板是一款拥有 4 个 Cortex-A53 核心,运行频率 1.8GHz;1 个 Cortex-M7 核心,运行频率 800MHz;此外还集成了一个 2.3 TOPS 的 NPU,大大加速机器学习推理。 全文所使用的开发平台均为与NXP官方合作的FS-IMX8…

深入浅出Kafka

这个主题 武哥漫谈IT ,作者骆俊武 讲得更好 一、什么是Kafka? 首先我们得去官网看看是怎么介绍Kafka的: https://kafka.apache.org/intro Apache Kafka is an open-source distributed event streaming platform. 翻译成中文就是&#xff1…

Git分布式版本控制软件

1.什么是git git是分布式版本控制软件。 软件:git是从别的地方下载下来安装到我们电脑上的软件。 版本控制:跟毕业论文一样,先写好版本1然后交给导师,导师说不行,然后再改为版本2,然后循环下去&#xff0…

Su+ELK实现网络监测(2)——ELK安装与配置

ELK安装配置文档一、环境准备基础配置二、Jdk1.8环境部署1、安装jdk2、编辑环境变量三、ElasticSearch部署1、安装2、修改文件所有者3、修改配置文件4、启动四、elasticsearch-head部署(可不安装,跳过)1、nodejs安装2、head插件安装3、修改he…

智慧工厂可视化合集,推动行业数字化转型

图扑软件基于 HTML5(Canvas/WebGL/WebVR)标准的 Web 技术,满足了工业物联网跨平台云端化部署实施的需求,以低代码的形式自由构建三维数字孪生、大屏可视化、工业组态等等。从 SDK 组件库,到 2D 和 3D 编辑,…

软化水处理知识总结

软化水除了广泛应用在饮用、浴室、厨房、洗衣等生活用水,和酒店、学校、写字楼、公寓、餐饮等商业用水的处理,还可用于锅炉、交换器、蒸发冷凝器、空调、直燃机等系统的补给水的软化。 那什么是软化水,和除盐水、纯水有什么区别?…

大学物理第四单元:刚体

1.刚体的定轴转动 思考:改变转动状态的因素 答:改变刚体运动状态的的因素有力的大小及力臂有关,力与力臂的乘积为力矩。 力臂:力到转轴的距离 简而言之,改变刚体运动状态的因素是力矩,与力和力臂有关。 …

谷粒学院项目笔记第一部分

1.环境搭建&#xff0c;准备工作 &#xff08;1&#xff09;创建数据库 &#xff08;2&#xff09;创建项目完整结构 &#xff08;3&#xff09;父工程springboot,子工程maven &#xff08;4&#xff09;父工程pom设置版本&#xff0c;添加pom #版本 <version>2.2.1.R…

11个AI写作软件工具!知名4A广告公司蓝标宣布停止文案外包!

AI的这场熊熊大火&#xff0c;终于还是烧到了广告界&#xff01; 2023年4月12日&#xff0c;是一个再普通不过的日子&#xff0c;但这一天会被很多人记住。不是因为席卷整个华北区的漫天黄沙&#xff0c;而是因为一封代表着AI势不可挡的决心和象征着一个行业巨变拉开序幕的邮件…

学会 制作极简搜索浏览器 —— 并将 ChatGPT 接入浏览器

前期回顾 Vue3 Ts Vite pnpm 项目中集成 —— eslint 、prettier、stylelint、husky、commitizen_0.活在风浪里的博客-CSDN博客搭建VIte Ts Vue3项目并集成eslint 、prettier、stylelint、huskyhttps://blog.csdn.net/m0_57904695/article/details/129950163?spm1001.2…