npm
嵌套结构(npm2)
在 npm2 中,npm 的处理依赖的方式很粗暴,直接采用递归的方式,严格的按照 package.json 结构以及子依赖包中的 package.json 结构将依赖安装到各自的 node_modules 中
举个🌰来说,npm-test项目依赖如下三个模块:
{
"name": "npm-test",
"dependencies": {
"buffer": "^5.4.3",
"ignore": "^5.1.4"
}
}
buffer
模块依赖了base64-js
和ieee754
模块
{
"name": "buffer",
"dependencies": {
"base64-js": "^1.3.1",
"ieee754": "^1.1.13"
}
}
执行 npm install 之后,我们会得到如下图的目录结构:
将其内部的依赖全部画出来会成为如下图:
这种方式的优劣式都非常的明显。优点就是 node_modules 的结构和 package.json 的结构是一一对应的,结构层级很明显,并且能够保证每次 install 的目录都是相同的。
试想一下,依赖的模块非常多,项目的 node_modules 会非常的庞大,嵌套会很深
从上图中,我们可以看出来嵌套结构的劣势
- 在不同层级的依赖中,可能引用了同一个模块,导致大量的模块冗余
- 在 windows 系统中,文件路径最大长度为260字符,嵌套层级过深可能导致不可预知的问题
💡 在 npm2 中,按照递归的方式,严格将 package.json 中的依赖安装到对应模块下。并不会处理某几个模块中的相同版本依赖,直接无脑生成对应树结构
扁平结构(npm3)
在此之后的 npm3 做了较大的更新,将 npm2 的嵌套结构改成了扁平结构
子依赖模块无关联
安装模块时,不管其是直接依赖模块还是子依赖的依赖模块,都优先安装在 node_modules 根目录下
执行 npm install 之后,会得到如下的目录结构
将其内部的依赖全部画出来会成为如下图:
💡 如果 package.json 中的依赖的子依赖无相同依赖,那么所有的依赖都会被扁平化到根目录的 node_modules 下
子依赖项依赖相同/兼容版本
修改 npm-test 的 package.json 文件,添加[email protected]
,websocket-util 也依赖了base64-js^1.3.0
{
"name": "npm-test",
"dependencies": {
"buffer": "^5.4.3",
"ignore": "^5.1.4",
"websocket-util": "1.0.0"
}
}
执行 npm install,会得到如下的目录结构:
将其内部的依赖全部画出来会成为如下图:
💡 如果 package.json 中的依赖的子依赖有相同或者兼容版本依赖,那么所有的依赖都会被扁平化到根目录的 node_modules 下
子依赖项的依赖不兼容
-
我们在项目 npm-test 中,又依赖了[email protected]
版本,修改 package.json:
{
"name": "npm-test",
"dependencies": {
"buffer": "^5.4.3",
"ignore": "^5.1.4",
"base64-js": "1.0.1",
}
}
执行 npm install 之后的目录结构如下:
将其内部的依赖全部画出来会成为如下图:
npm-test 直接依赖的[email protected]
放在了根目录的 node_modules下,buffer 所依赖的与其不兼容,就放在自身的 node_modules下
-
修改 package.json,在其中添加[email protected]
,它依赖[email protected]
;[email protected]
依赖base64-js^1.3.0
,buffer依赖base64-js^1.3.1
{
"name": "npm-test",
"dependencies": {
"bops": "1.0.1",
"buffer": "^5.4.3",
"ignore": "^5.1.4",
"websocket-util": "1.0.0"
}
}
执行 npm install,会得到如下的目录结构:
将其内部的依赖全部画出来会成为如下图:
会发现[email protected]
被提取到了第一级的 node_modules 上,而 buffer/websocket-util 依赖的依旧挂在它自己的 node_modules 下
package.json 的安装顺序是按着字母顺序来的,首先 bops 的底层依赖都会被优先提出来,所以 node_modules 下会先有[email protected]
,到了处理 buffer/websocket-util 的底层依赖时,发现已经存在 base64-js 且不兼容就会放到它自身的 node_modules 下
-
我们在往 package.json 中加入[email protected]
它依赖base64-js^1.5.1
的版本
{
"name": "npm-test",
"dependencies": {
"ag-psd": "14.3.6",
"bops": "1.0.1",
"buffer": "^5.4.3",
"ignore": "^5.1.4"
}
}
执行 npm install 之后,得到目录结构如下:
将其内部的依赖全部画出来会成为如下图:
能够发现,这次是[email protected]
被提取到了根目录下的node_modules下,buffer 的base64-js 能够和它兼容,所以 buffer 的 node_modules 下不再存在依赖,然而 bops 依赖的 base64-js 不兼容,所以会挂在自身的 node_modules 下
💡 子依赖项的依赖不兼容的情况下,底层会通过localeCompare的方法对依赖进行一个排序,字典序靠前的 npm 包的底层依赖会优先被提取出来,放到根目录下的 node_modules 中,之后如果发现不兼容的依赖,则继续采用 npm 2 的处理方式,都会放在自身的 node_modules 下
❓通过上面这几个例子,能够发现 npm3 在解决了一些问题的同时,也带来新的问题。
npm3 仿佛解决了 npm2 的冗余问题,但是也没有完全解决。例如上面的例子,[email protected]
先被提出来,如果后面的包依赖的 base64-js 和1.0.2版本不兼容,就会导致每个子依赖的 node_modules 都会存在 base64-js 包,又出现了 npm2 的冗余问题
package-lock.json
在之前我们讲了 package.json 文件,在 npm5 中,推出了 package-lock 文件,它就是每个依赖项的列表,当前安装的版本,模块位置,验证模块完整性的哈希,以及对应的包列表/依赖项列表。其实也是和 node_modules 一一对应的,项目目录下存在 package-lock 可以让每次安装生成的依赖目录结构保持相同
{
"name": "npm-test",
"dependencies": {
"buffer": "^5.4.3",
"ignore": "^5.1.4"
}
}
上述 package.json 执行 npm install 之后得到的 lock 文件如下:
{
"name": "npm-test",
"dependencies": {
"base64-js": {
"version": "1.5.1",
"resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
"integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA=="
},
"buffer": {
"version": "5.7.1",
"resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz",
"integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
"requires": {
"base64-js": "^1.3.1",
"ieee754": "^1.1.13"
}
},
"ieee754": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
"integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA=="
},
"ignore": {
"version": "5.2.0",
"resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.0.tgz",
"integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ=="
}
}
}
resolved: 依赖包的位置URI
integrity: 验证模块完整性的哈希
当我们的项目中已经存在 package-lock.json 之后,将以该文件为主进行解析安装指定版本依赖包,而不是使用 package.json 来解析和安装模块。因为 package-lock 为每个模块都指定了版本/位置/完整性哈希,所以每次创建的安装都是一样的。和使用的设备无关,每次都能给到相同的结果。
该图为第一次 install 和有了 package-lock 文件之后 install 时间的对比。在依赖少的情况下并不明显,依赖越多时间差会更加明显
在 npm5.0.x的版本中,不管 package.json 中依赖是否有更新,都会以 package-lock 文件为第一安装依赖
在 npm5.1.0之后的版本中,如果 package.json 中的依赖项有更新,install 时会无视 package-lock 直接去下载新版本依赖,然后在更新 package-lock 文件
其实上面的两种方案都存在对应的问题,因此在5.4.2的版本之后,更改了规则
- 根据 package.json 文件,运行 install 会生成对应的 package-lock 文件,package-lock 文件指明了直接依赖版本和间接依赖版本
- 如果 package 和 lock 文件中依赖版本兼容,即使 package 中有新版本,执行 install 的时候也会根据 lock 文件下载
- 如果两个文件中的版本不兼容,那么执行 install 的时候会把 lock 文件更新到兼容package.json 的版本
缓存
在我们执行 install/update 时,除了将依赖包安装在 node_modules 目录之外,还会在本地缓存目录缓存一份
执行npm config get cache
命令可以查询到缓存目录,Mac默认是在用户主目下.npm/_cacache
content-v2 目录用于存储 tar 包的缓存,而 index-v5 目录用于存储 tar 的 hash 值
查找缓存的步骤
-
根据 lock 文件中的integrity
和resolved
字段,通过pacote:range-manifest:{url}:{integrity}
构建出key
例如: pacote:version-manifest:https://registry.npmjs.org/base64-js/-/base64-js-1.0.2.tgz:sha1-R0IRyV5s8qVH20YeT2d4tR0I+mU=
-
通过 SHA256 算法加密得到一个 Hash 值
49c780b41697e5f2aa0641d954ec662e78d57096456d0108d1a84ba538306284
-
加密之后的 Hash 值的前四位就是 index-v5 目录的下两级
_cacache/index-v5/49/c7/…
-
通过路径找到对应的索引文件,在文件中找到对应的 _shasum 字段
474211c95e6cf2a547db461e4f6778b51d08fa65
-
上述字符串表明的就是缓存包的位置,对应的前四位为目录的下两级,执行tar -xvf file
就能解压文件
_cacache/content-v2/47/42/…
文件完整性
上文提到过几次文件完整性,那具体什么是文件完整性?
在下载依赖包之前,能够拿到 npm 对该依赖包计算的 hash 值。执行 npm info 命令,shasum
就是 hash
用户下载依赖包到本地之后,要确定在下载的过程中没有出现错误,所以在下载完成之后在本地计算一次文件的 hash 值。如果两个 hash 值相同才能够确保下载的依赖是完整的;如果不同则需要重新下载
npmrc文件
.npmrc 文件可以理解成为 npm running configuration,即 npm 运行时配置文件。
npm 的作用就是帮助开发者安装需要的依赖包,但是要从哪里下载?这是可以在.npmrc中进行配的。
在我们安装包的时候,npm按照如下顺序读取这些配置文件:
- 项目配置文件:你可以在项目的根目录下创建一个.npmrc文件,只用于管理这个项目的npm安装。
- 用户配置文件:在你使用一个账号登陆的电脑的时候,可以为当前用户创建一个.npmrc文件,之后用该用户登录电脑,就可以使用该配置文件。可以通过 npm config get userconfig 来获取该文件的位置。
- 全局配置文件: 一台电脑可能有多个用户,在这些用户之上,你可以设置一个公共的.npmrc文件,供所有用户使用。该文件的路径为: $PREFIX/etc/npmrc,使用 npm config get prefix 获取$PREFIX。如果你不曾配置过全局文件,该文件不存在。
- npm内嵌配置文件:最后还有npm内置配置文件,基本上用不到,不用过度关注。
整体流程
- 检查 .npmrc 文件,优先级为:项目级 > 用户级 > 全局级 > npm内置
- 检查有无 lock 文件
- 无 lock 文件
- 从 npm 远程仓库获取包的信息
- 根据 package.json 构建依赖树,构建过程如下
- 首先确定首层依赖模块
dependencies/devDependencies/optionalDependencies
,工程本身是整棵依赖树的根节点,每个首层模块都是根节点下的一个子树,此时会开启多进程从每个首层依赖模块开始逐步寻找更深层级的节点
- 这一步只是确定逻辑上的依赖树,并非真正的安装,之后根据这个依赖结构去下载或拿到缓存中的依赖包
- 模块扁平化
在上一个步骤的到的是一个完整的依赖树,包含了大量的重复模块,在npm3后就开启了扁平化操作,会遍历所有的节点(有广度遍历的感觉)将模块逐个放到根节点下,有重复模块时就丢弃;不兼容时则放到当前节点,不做改变。
- 在缓存中依次查找依赖树中的依赖
- 不存在缓存
- 从 npm 远程仓库下载包
- 校验包的完整性
- 校验不通过重新下载
- 校验通过,将下载的包复制到 npm 缓存目录;将下载的包按照依赖结构解压到 node_modules
- 存在缓存
- 将缓存按照依赖结构解压到 node_modules
- 生成 lock 文件
- 有 lock 文件
- 检查 package.json 中的依赖版本和 lock 文件是否兼容
- 如果兼容,直接跳过获取包信息、构建依赖树过程,开始在缓存中查找包信息,后续过程相同
- 如果不兼容,从远程获取包信息,后续过程相同
Yarn
yarn 的发布时间是在2016年,那时候的 npm 还处于v3阶段,还没有 package-lock.json 文件,存在的问题上文也提到过。yarn 就在此时诞生了
yarn 也是采用的 npm-v3 扁平化结构来管理包依赖,安装完成之后会生成一个 yarn.lock 文件
执行与上述一样的 package.json,得到如下的 yarn.lock 文件
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
# yarn lockfile v1
base64-js@^1.3.1:
version "1.5.1"
resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a"
integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==
buffer@^5.4.3:
version "5.7.1"
resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0"
integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==
dependencies:
base64-js "^1.3.1"
ieee754 "^1.1.13"
ieee754@^1.1.13:
version "1.2.1"
resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352"
integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==
ignore@^5.1.4:
version "5.2.0"
resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.0.tgz#6d3bac8fa7fe0d45d9f9be7bac2fc279577e345a"
integrity sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==
和 npm 产生的 lock 文件还是比较相似的,也有一些区别
- package-lock.json 是 json 格式,yarn.lock 使用一种自定义格式
- 所有依赖,不管是项目声明的依赖,还是依赖的依赖,都是扁平化管理
- 依赖的版本是由所有依赖的版本声明范围确定的,具备相同版本声明范围的依赖归结为一类,确定一个该范围下的依赖版本。如果同一个依赖多个版本共存,那么会并列归类
- 相比 npm,Yarn 一个显著区别是 yarn.lock 中子依赖的版本号不是固定版本。 也就是说单独一个 yarn.lock 确定不了 node_modules 目录结构,还需要和 package.json 文件进行配合
yarn.lock 是扁平化的,即使有相同的包也会带有版本号并列在一级;而 package-lock 是和 node_modules 一一对应的嵌套结构
所以 yarn.lock 即使有确定的版本号也没法得到对应 node_modules 的目录结构所以需要结合 package.json
yarn install 的过程,可以在终端看到由 emoji 图标装饰的四个步骤
- Resolving packages(解析包): 整合依赖信息
- Fetching packages(获取包): 获取依赖包到缓存中
- Linking dependencies(连接依赖): 复制依赖到 node_modules
- Building fresh packages(构建安装): 执行 install 阶段的 scripts
pnpm
什么是pnpm
其官网这样说:
Fast, disk space efficient package manager 快速的,节省磁盘空间的包管理工具
pnpm 本质上就是一个包管理器,在这一点和 npm/yarn 没有区别,但是其两个优势在于:
-
包安装数度极快
官方 benchmarks 对 npm、pnpm、yarn、yarnPnP 对多个情景下的性能基准测试,涵盖了很多使用场景
-
磁盘空间利用非常高效
使用 yarn 安装依赖时的node_modules
使用 pnpm 安装依赖时的 node_modules
pnpm 内部使用基于内容寻址的文件系统来存储磁盘上所有的文件,该文件系统出色的地方在于
- 不会重复安装同一个包。在 npm/yarn 中,我们所有的项目(10个)有依赖了 dt-common,dt-common 会被安装10次,dt-common 这个包在磁盘中就会有10个地方写入该代码。但是在 pnpm 中只会被安装一次,磁盘中只有一个地方会写入,后面再次使用都直接使用硬连接hardlink
-
支持monorepo
-
相对安全
前置知识-软链接和硬链接
在 Linux 中有两种链接方式:
- 硬连接 hard link
- 软链接 soft link,又称为符号链接 symbolic link
不论是硬链接或软链接都不会将原本的源文件复制一份,只会占用非常少量的磁盘空间
inode
每个文件都有唯一一个 inode,它包含了文件元信息,在访问文件时,对应的元信息会被 copy 到内存去实现文件的访问
使用stat xxx
能够查到对应的 inode
hard link
硬链接可以理解成为一个相互的指针,创建的 hardlink 指向源文件的 inode,系统并不为会其分配新的 inode
硬链接不管有多少个,都会指向用一个 inode 节点,这等同于修改源文件或者链接文件时都会被同步修改(感觉和JS对象引用地址略微相似
每新建一个 hardlink 会把节点连接数增加,只要节点的链接数非零,文件就一直存在,不管你删除的是源文件还是 hradlink。只要有一个存在,文件就存在
soft link
软链接可以理解为单向指针,一个独立的文件且拥有独立的 inode,永远指向源文件
修改源文件内容,软链接内容也会改变。当删除源文件时,访问软链接会报错No such file or directory
软硬链接对node寻包的影响
可以使用ln
命令对文件和目录在另一个位置建立一个同步的链接
语法: ln [参数][源文件或目录][目标文件或目录]
在某个项目中新建两个文件夹 one/two,目录结构如下:
one/index.js 中内容如下:
const base64 = require("base64-js")
console.log(base64)
现在运行当前 one/index.js 一定会报错,因为我们的 one 文件下不存在 base64-js 依赖
分别在 two 文件下创建 one/index.js 的硬链接/软链接文件
hard.js 是一个硬链接,soft.js 是一个软链接
我们分别执行 node hard.js/node soft.js
发现硬链接是可以正常运行的,但是软链接不行。这是因为硬链接会从链接到的位置开始查找依赖,而软链接会从文件原始位置开始查找依赖。
软链依赖目录
上述的例子讲清楚了 node 处理软硬链接的不同之处,那我们将 two/node_modules 软链接到 one 的目录下,one/index.js 和 two/soft.js 能否正常运行呢🤔
ln -s ../two/node_modules ./one
对应执行one/index.js 和 two/soft.js。哇哦,work well
通过上面的示例,我们发现即使当前 node_modules 是一个软链接也能够作为依赖被查找到,说明软链可以将其他地方的目录增加到依赖查找路径中。
依赖管理
{
"name": "npm-test",
"dependencies": {
"buffer": "^5.4.3"
}
}
在我们的 npm-test 项目中执行pnpm i
控制台会有如下输出:
能够发现在 Progress 中,能够明确有多少包被复用和重新下载了多少包
并且当我们运行pnpm install
进行 node_modules 安装的时候,会使用软链接 & 硬链接的方式来节省磁盘空间 & 提升安装效率
pnpm的node_modules目录结构(软链接的使用)
上述安装得到如图的 node_modules:.pnpm 文件 & package.json描述的其他文件
- .pnpm 目录:存放了所有实际安装的包,它里面全部都是我们项目所需要的依赖,唯一不同的它们是来自 store 目录的硬链接
- 其他文件:package.json中声明的包,只是生成一个软链接,实际指向.pnpm中安装的包
因为我们只依赖项中直定义了buffer
,所以它是唯一一个你的应用必须拥有访问权限的包(想了解为什么要对依赖项做严格控制,点击查看)
pnpm 这种依赖管理的方式也很巧妙地规避了非法访问依赖
的问题,也就是只要一个包未在 package.json 中声明依赖,那么在项目中是无法访问的
但是在 yarn/npm 中因为存在依赖提升问题,buffer 依赖 base64-js,因此 node_modules 中存在 base64-js 的包,我们在项目中可以直接引用。这就是为什么 pnpm 的包管理更为安全的原因
那 buffer 目录下都有些啥呢?发现 buffer 下根本没有 node_modules,它的依赖项都放在哪里了?目录结构第一个诀窍,buffer 是一个软链接,在 node 解析依赖的时候,它使用这些依赖的真实位置,因此在 buffer 下不用保留依赖。
那你又会问 buffer 的真实地址在哪里?🧐
嚯,真实地址在这里: node_modules/.pnpm/[email protected]/node_modules/buffer
所以我们现在知道了.pnpm/
文件夹的用途。.pnpm/
以平铺的形式储存着所有的包,所以每个包都可以在这种命名模式的文件夹中被找到:node_modules/.pnpm/<name>@<version>/node_modules/<name>
使用这种平铺的结构避免 npm2 嵌套结构引起的长路径问题,和 npm3/yarn 的扁平化结构不同的是它保留了包之间的相互隔离
点开 buffer 的真实地址,再一次发现自己又被骗了,真实地址下面还是没有 node_modules
目录结构第二个诀窍来了,包的依赖项与依赖包的实际位置位于同一目录级别,buffer 的依赖不在.pnpm/[email protected]/node_modules/buffer/node_modules
,而是在.pnpm/[email protected]/node_modules/
。buffer 的所有依赖都被软链接到了node_modules/.pnpm
的对应目录
到了这里你可能会问,为什么要把子依赖放在统一层级🤔
假如我们有两个包,foo 和 bar 相互依赖,使用同一层级安装的目录如下,这时 foo 和 bar 的软链接不是一个循环
.pnpm
[email protected]
node_modules
bar
foo --> ../../[email protected]/node_modules/foo
[email protected]
node_modules
foo
bar --> ../../[email protected]/node_modules/bar
假设 bar/foo 的依赖放在它的子文件中,如下,这时 foo/bar 的软链接就是一个循环
.pnpm
[email protected]
node_modules
bar
node_modules
foo --> ../../[email protected]/node_modules/foo
[email protected]
node_modules
foo
node_modules
bar --> ../../[email protected]/node_modules/bar
pnpm下node_modules存在的意义
buffer 中有 base64/ieee754 两个包,假设在 base64 的 dependencies 没有声明ieee754,那我们在 base64/index 使用requie('ieee754')
时,他就是一个幽灵依赖。会先寻找 base64 上级的node_modules,若找到直接使用;否则直接往上级寻找 node_modules,如果上级也没有的话,当前的使用就会有问题
所以pnpm会将所有的依赖都提升到.pnpm/node_modules下,保证幽灵依赖可以被找到,作为兜底方案discussions
硬链接使用
在使用 pnpm 安装时,pnpm 会将依赖存储在~/.pnpm-store/v3
目录下。在同一台电脑下,下一次安装的时候 pnpm 会先检查 store 目录,如果有找到所需的依赖,则会通过硬链接放到我们的项目中去,而不是重新安装依赖
因为 pnpm 安装依赖时会将依赖存储在 store 目录下,该目录能够使不同的项目之间共享依赖。例如 dt-data-api 项目中中安装了 dt-common/dt-react-component 等依赖,当我们在dt-easy-index 项目中也用了该两个依赖时,会重复使用 store 下的依赖。而 yarn/npm 每次都会重新安装依赖
🤔那么问题来了,为什么在这里要使用硬链接,而不是直接创建到全局存储的符号链接
通过上文我们知道 node 对软硬链接查询位置的处理不同。假设我们有如下的两个项目:
foo
"react": "16.8.0",
"dt-react-component": "1.0.0"
bar
"react": "17.0.0",
"dt-react-component": "1.0.0"
在dt-react-component
里面,会使用import React from 'react'
,假设foo/node_modules/dt-react-component
和bar/node_modules/dt-react-component
都使用软链接到./pnpm-store/v3
,那他们都会使用同一个版本的 react,这不是我们所期望的。所以使用硬链接,在dt-react-component
中在使用import React from 'react'
时,会根据当前的位置去寻找依赖,这样使得 foo 和 bar 可以依赖不同的react的版本
pnpm的依赖管理图
参考链接