详细讲解npm install命令执行,都干了哪些事情?
liuian 2025-05-03 15:15 4 浏览
详细讲解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 中的错误记录。
相关推荐
- Docker 47 个常见故障的原因和解决方法
-
【作者】曹如熙,具有超过十年的互联网运维及五年以上团队管理经验,多年容器云的运维,尤其在Docker和kubernetes领域非常精通。Docker是一种相对使用较简单的容器,我们可以通过以下几种方式...
- 电脑30个快问快答,解决常见电脑问题
-
1.强行关机/停电对电脑有影响吗?答:可能损坏硬盘(机械硬盘风险高)、未保存数据丢失,偶尔一次影响小,但频繁操作会缩短硬件寿命。2.C盘满影响速度吗?答:会!系统运行需C盘空间缓存临时数据,空间不...
- 使用Tcpdump包抓取分析数据包的详细用法
-
TcpDump可以将网络中传送的数据包的“头”完全截获下来提供分析。它支持针对网络层、协议、主机、网络或端口的过滤,并提供and、or、not等逻辑语句来帮助你去掉无用的信息。tcpdump就是一种...
- 电脑启动不了(BootDevice Not Found Hard Disk-3F0)解决方案
-
HP品牌机,开机启动不了,黑屏,开机取下主板电池恢复BIOS后,开机显示找不到启动盘。一、按F2键进入BIOS,出现硬盘内存检测界面的话,直接退出。就会出现这个界面,光标键向下,选择BIOSSetu...
- 电脑开机黑屏别慌!快码住!起底维修老师傅不能说的秘密
-
按下开机键却只收获黑屏大礼包?那些神秘的英文提示、刺耳的蜂鸣声,其实是电脑在给你发送求救信号!从按下电源到进入桌面的12秒里,你的电脑经历了史诗级的硬件自检与系统加载,今天我们就破译这段“摩斯电码”。...
- 电脑启动故障为何总要先看BIOS?新手必读的关键知识解析
-
最近在帮朋友们解答电脑无法正常开机的问题时,发现大家经常收到一句高频建议:“先检查BIOS”。对不少普通用户而言,BIOS依然是个神秘的存在。那么,BIOS到底是什么?电脑出现哪些故障会与它相关呢?本...
- Windows 11 KB5053598更新:安全补丁还是系统噩梦?
-
2025年3月11日,微软发布了Windows1124H2的强制性更新KB5053598,作为“周二补丁日”(PatchTuesday)的一部分。然而,这款本应提升系统安全性的更新却引发了广泛的...
- 飞牛OS入门安装遇到问题,如何解决?
-
之前小编尝试了用旧电脑装飞牛OS安装之前特意查了一些硬件要求飞牛OS目前支持主流的x86架构硬件主机需能连网线飞牛OS暂时不支持只有无线网卡的安装貌似很多小伙伴在一开始安装就卡住了那今天咱们汇总分...
- 几种常见的电脑开机黑屏显示白色英文字母解决方法
-
当电脑开机出现黑屏并显示白色英文字母时,通常表示系统启动过程中遇到了错误。以下是几种常见原因及对应的解决方法,按照排查顺序整理:一、检查外接设备与硬件连接可能原因:外接U盘、移动硬盘等未拔出,或内部硬...
- 电脑启动出现问题,为什么都要先检查BIOS?
-
【ZOL中关村在线原创技巧应用】最近在回答问题的时候,总会发现很多朋友都在问“电脑无法正常开机怎么办?”这样类似的问题,而许多DIY大佬的回复总会出现一条高频建议“先检查BIOS”。但对于许多普通用户...
- 教你怎么用JavaScript检测当前浏览器是无头浏览器
-
什么是无头浏览器(headlessbrowser)?无头浏览器是指可以在图形界面情况下运行的浏览器。我可以通过编程来控制无头浏览器自动执行各种任务,比如做测试,给网页截屏等。为什么叫“无头”浏览器?...
- 12个高效的Python爬虫框架,你用过几个?
-
实现爬虫技术的编程环境有很多种,Java、Python、C++等都可以用来爬虫。但很多人选择Python来写爬虫,为什么呢?因为Python确实很适合做爬虫,丰富的第三方库十分强大,简单几行代码便可实...
- 运维的报表之路,用 node.js 轻松发送 grafana 报表
-
在运维过程中,无论是监控还是报表,都会有一些通过邮件发送图表的需求,由于开源的zabbix,grafana和kibana等并不完全具有“想发送哪儿就发送哪儿”的图片生成功能,在grafana...
- C#基于浏览器内核的高级爬虫(c#爬取网页内容)
-
基于C#.NET+PhantomJS+Sellenium的高级网络爬虫程序。可执行Javascript代码、触发各类事件、操纵页面Dom结构、甚至可以移除不喜欢的CSS样式。很多网站都用Ajax动态加...
- 如何优化一个秒杀项目?(秒杀实现思路)
-
问题1:使用jmeter性能压测,定位瓶颈代码步骤流程:线程组--->Http请求--->查看结果树--->聚合报告tips:host的文件--->优先调用映射,减少DNS的时...
- 一周热门
-
-
Python实现人事自动打卡,再也不会被批评
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
python使用fitz模块提取pdf中的图片
-
《人人译客》如何规划你的移动电商网站(2)
-
Jupyterhub安装教程 jupyter怎么安装包
-
- 最近发表
- 标签列表
-
- 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)
- table.render (33)
- uniapp textarea (33)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- react-admin (33)
- vscode切换git分支 (35)
- vscode美化代码 (33)
- python bytes转16进制 (35)