详细讲解npm install命令执行,都干了哪些事情?
liuian 2025-05-03 15:15 58 浏览
详细讲解npm install命令执行,都干了哪些事情?
一、依赖解析阶段
1. 读取 package.json
- 目的:确定项目需要的 直接依赖(dependencies 和 devDependencies)。
- 关键字段:
- 输出:生成一个 初始依赖列表,包含所有直接声明的包及其版本约束。
2. 处理 package-lock.json 或 npm-shrinkwrap.json
- 目的:确保依赖树的 确定性(不同环境安装相同版本)。
- 优先级:如果存在这些文件,npm 会优先使用其中锁定的 精确版本号,而非 package.json 的版本范围。
- 例外:当使用 npm install <package> 手动安装新包时,会更新 package-lock.json。
3. 依赖版本解析
- 算法:npm 使用 语义化版本(SemVer) 解析依赖树:^4.17.20:允许安装 4.x.x 的最新版本,但不包括 5.0.0。~4.17.20:允许安装 4.17.x 的最新版本。
- 冲突处理:如果多个依赖要求同一个包的 不兼容版本,npm 会尝试找到满足所有条件的版本,否则抛出 ERESOLVE 错误(需手动解决)。
二、依赖树构建
1. 递归解析嵌套依赖
- 过程:从直接依赖开始,逐层解析每个包的 dependencies,形成一棵 依赖树。
- 示例:
2. 扁平化处理(Dedupe)
- 目的:减少重复安装,优化 node_modules 结构。
- 策略:将不同层级但版本兼容的依赖 提升到顶层。
- 冲突:如果同一包的不同版本无法兼容,会 分别安装到各自父目录 下。
三、安装策略
1. 缓存检查
- 路径:~/.npm/_cacache(存储所有下载过的包)。
- 机制:计算包的 完整性哈希(如 sha512),检查缓存是否存在。若存在,直接从缓存 硬链接 到 node_modules(节省下载时间)。
2. 下载缺失包
- 源:从 registry(默认 https://registry.npmjs.org/)下载未缓存的包。
- 镜像配置:通过 .npmrc 或 npm config set registry 切换镜像源(如淘宝源)。
3. 解压与文件处理
- 步骤:
- 下载的 .tgz 压缩包解压到 node_modules。
- 创建 package.json 中指定的 二进制文件链接(如 bin 字段)。
- 执行 生命周期脚本(如 preinstall、postinstall)。
四、生命周期脚本
1. 执行顺序
2. 典型场景
- 原生模块编译:如 node-sass 在 install 阶段调用 node-gyp 编译 C++ 代码。
- 环境检查:某些包会检查 Node.js 或操作系统版本,不满足条件时抛出错误。
五、生成或更新 package-lock.json
- 目的:记录 精确的依赖树结构 和每个包的版本。
- 更新时机:当 package.json 变更时(如添加新依赖)。当依赖的远程版本更新且符合 SemVer 约束时。
- 重要性:提交到版本控制系统,确保团队协作和 CI/CD 环境的一致性。
六、特殊安装模式
1. npm ci
- 特点:严格依赖 package-lock.json,删除 node_modules 后重新安装。
- 适用场景:持续集成环境,确保安装结果与锁文件完全一致。
2. npm install --production
- 行为:仅安装 dependencies,跳过 devDependencies。
- 适用场景:生产环境部署,减少不必要的开发依赖。
七、常见问题与优化
1. 依赖冲突
- 表现:ERESOLVE unable to resolve dependency tree。
- 解决:
2. 加速安装
- 缓存利用:避免频繁删除 node_modules,利用 npm cache verify 清理无效缓存。
- 镜像源:使用国内镜像或企业私有仓库。
3. 锁定依赖版本
- 精确指定:在 package.json 中使用固定版本(如 4.17.20 而非 ^4.17.20)。
- 禁用自动更新:
总结
npm install 是一个复杂的流程,涵盖依赖解析、树构建、缓存管理、脚本执行等多个环节。理解其内部机制有助于:
- 快速诊断安装失败的原因(如网络问题、版本冲突)。
- 优化项目依赖结构,减少 node_modules 体积。
- 确保团队和部署环境的一致性。
遇到问题时,可结合 npm install --verbose 输出详细日志,或检查 ~/.npm/_logs 中的错误记录。
相关推荐
- 新冠疫苗接种查询(新冠疫苗接种查询不到了)
-
银川市疫苗接种信息在“我的宁夏”app里查询。打开“我的宁夏”app,打开“健康码”选项,然后打开“疫苗接种查询”,就可以查询到你的疫苗接种信息了。如果你接种了吗两次疫苗,你的健康码就会镶嵌了金边,只...
- cdr格式文件怎么打开(cdr格式用什么可以打开)
-
cdr是什么文件?cdr文件用什么打开?cdr文件是CorelDraw图像制作软件标准的输出格式,与photoshop(PS)图片设计软件类似都属于图片设计软件,需要打开cdr文件我们就需要先了解C...
- 微软win7系统下载(微软官网window7下载)
-
因为你是按照名称排列的,试着修改一下排序方式,方法是:在桌面空白处右键,排序方式,修改日期这是因为大部分电脑用的都是GHOST系统,该系统在安装后都会向注册表写入自己的“私人信息”,如:风林火山GH...
- xp10下载(Xp10下载破解版)
-
下载win10系统所需要的时间取决于很多因素。首先就是系统本身的大小,一般情况下,纯净版的系统在2.3个g左右,而有一些定制版的系统大小就不好说了,但是删减版的系统可以做到1.7个g左右,系统本身文件...
- 最强神级选择系统(无敌之最强神级选择系统)
-
8.自由高达《机动战士高达SEED》 7.V高达《机动战士高达V》 6.00高达《机动战士高达00》 5.独角兽高达《机动战士高达UC》 4.倒A高达《倒A高达》 3.V2高达《机动战士高...
- 惠普服务器售后电话客服热线
-
售后官网:https://support.hp.com/cn-zh/check-warranty该网站为惠普全新一站式服务支持平台,小惠将为您提供专业的技术解答和产品导购。同时提供全面的自助解决方案...
- win10改成win7界面(win10改为win7界面)
-
1、首先我们打开你的电脑,选择打开左下角的开始菜单。2、打开后就选择设置按钮。3、打开之后就是这个页面往下拉,你就会看见更新,就是红圈圈里面,打开它。4、打开之后就是这个页面。5、然后我们在左侧选择恢...
- win10系统怎么恢复系统还原(win10系统恢复怎么操作)
-
1、开机不断点击F8键,进入系统操作选单,选“最后一次正确配置”,重启电脑,看能否解决。2、开机不断点击F8键,进入系统操作选单,选“安全模式”,如能成功进入,依次单击“开始”→“所有程序”→“附件”...
- 怎么把电脑c盘格式化(怎么把c盘格式化吗)
-
1.在进行c盘格式化之前,必须备份重要的数据,以免数据丢失。2.在格式化c盘之前,确保所有的应用程序和系统文件都已经备份。否则,在进行格式化之后,将需要重新安装操作系统和所有的应用程序。3.如果...
- 中国移动赠送的wifi6路由器(192.168.10.1中国移动路由器)
-
移动tclwifi6路由器很好的。WiFi6路由器远强于WiFi5路由器,值得购买性价比不错。家庭生活中网速慢最主要的原因并非速率,而是穿墙效果。坐在离路由器很近的地方,手机网速一般不会有问题。...
- win官网网址(win官方网站)
-
具体方法如下:打开电脑,在浏览器地址栏输入window,就会进入win中文官网,然后再寻找自己需要的软件,升级包等就可以了。下面是windows官网网址:1http://www.microsoft.c...
- wifi网址登录入口(wifi网址登录入口有哪些)
-
wifi网页登录入口是网关地址192.168.1.1.1.首先检查路由器线路连接(参考前面手机设置路由器部分)2.将电脑Tnternet协议(TCP/IP)属性设置为“自动获取IP地址”和“自...
- 没有驱动程序怎么安装打印机
-
(1)启动电脑进入Windows操作系统,在桌面上单击开始-设置-打印机。(2)用鼠标右单击安装好的打印机图标,在弹出的右键菜单中选择属性,然后单击打印测试页按钮。(3)此时打印机会打出一页信息,从这...
- 一周热门
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
