npm 的 lock 机制解析 - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
QZhan
V2EX    分享发现

npm 的 lock 机制解析

  •  1
     
  •   QZhan 2019-06-20 22:24:46 +08:00 2456 次点击
    这是一个创建于 2316 天前的主题,其中的信息可能已经有所发展或是发生改变。

    npm 是什么

    npm 是一个包管理工具,开源作者可以把开源包发布在平台上供其他人下载使用。前的同学基本都使用过 npm,这里就不做过多介绍。日常工作中 npm 的主要用途就是根据项目的 package.json 使用 npm install 去安装依赖。

    npm install 可以说是我们使用最频繁的一个指令。在 npm5 版本之前,npm install 会根据 package.json 指定的依赖版本去进行安装。但往往 package.json 中指定的是一个版本范围,例如:

    "dependencies": { "packageA": "^2.0.0" }, 

    以上这个 ^2.0.0 指定的范围是版本号大于等于 2.0.0 且大版本号为 2。即 2.6.10 这个是符合的,而 3.0.0 和 1.0.0 这种是不符合的。

    这样的范围指定会导致一个问题:A 新建了一个项目,生成了上面这份 package.json 文件,但 A 安装依赖的时间比较早,此时 packageA 的最新版本是 2.1.0,该版本与代码兼容,没有出现 bug。后来 B 克隆了 A 的项目,在安装依赖时 packageA 的最新版本是 2.2.0,那么根据语义 npm 会去安装 2.2.0 的版本,但 2.2.0 版本的 API 可能发生了改动,导致代码出现 bug。

    这就是 package.json 会带来的问题,同一份 package.json 在不同的时间和环境下安装会产生不同的结果。

    理论上这个问题是不应该出现的,因为 npm 作为开源世界的一部分,也遵循一个发布原则:相同大版本号下的新版本应该兼容旧版本。即 2.1.0 升级到 2.2.0 时 API 不应该发生变化。

    但很多开源库的开发者并没有严格遵守这个发布原则,导致了上面的这个问题。

    lock 机制

    一个新的事物的诞生都是为了解决一个历史问题

    基于这种状况,npm5 推出了 lock 机制。在使用 npm5.0.0 之后的版本时,npm install 后会自动生成 package-lock.json 文件,该文件记录了当前这次 install 所安装的依赖版本号。

    例如当 package.json 的依赖如下:

    "dependencies": { "vue": "^2.0.0" }, 

    install 后自动生成的 package-lock.json 会指定安装 vue2.6.10 版本(当前最新)

    "dependencies": { "vue": { "version": "2.6.10", "resolved": "https://registry.npm.taobao.org/vue/download/vue-2.6.10.tgz", "integrity": "sha1-pysaQqTYKnIepDjRtr9V5mGVxjc=" } } 

    package-lock.json 相当于本次 install 的一个快照,它不仅记录了 package.json 指明的直接依赖的版本,也记录了间接依赖的版本。

    如果我们想在不同环境和不同时间下每次 install 时安装相同版本的依赖,我们就可以把 package-lock.json 带上。

    当 package.json 和 package-lock.json 同时存在时,npm install 会去检测 package-lock.json 指定的依赖版本是否在 package.json 指定的范围内。如果在,则安装 package-lock.json 指定的版本。如果不在,则忽略 package-lock.json,并且用安装的新版本号覆盖 package-lock.json。

    举个例子:

    // package.json "dependencies": { "vue": "^2.0.0" } // package-lock.json "dependencies": { "vue": { "version": "2.1.0", "resolved": "https://registry.npm.taobao.org/vue/download/vue-2.1.0.tgz", "integrity": "sha1-KTuj76rKhGqmvL+sRc+FJMxZfj0=" } } 

    这种情况下 package-lock.json 指定的 2.1.0 在^2.0.0 指定的范围内,npm install 会安装 vue2.1.0 版本。

    // package.json "dependencies": { "vue": "^2.2.0" } // package-lock.json "dependencies": { "vue": { "version": "2.1.0", "resolved": "https://registry.npm.taobao.org/vue/download/vue-2.1.0.tgz", "integrity": "sha1-KTuj76rKhGqmvL+sRc+FJMxZfj0=" } } 

    这种情况下 package-lock.json 指定的 2.1.0 不在^2.2.0 指定的范围内,npm install 会按照^2.2.0 的规则去安装最新的 2.6.10 版本,并且将 package-lock.json 的版本更新为 2.6.10 。

    值得注意的是 npm5 一发布时并不是采取这种 install 逻辑,在 npm5.0 到 npm5.6 之间 install 的逻辑发生了多次变更,而在 npm5.6 之后一直沿用当前这种逻辑。

    npm ci

    npm5 之后的 lock 机制满足了要求锁版本的开发者们的需要,我们只需要拿到一份 package-lock.json 就可以知道要安装的依赖的具体版本号。但细心的同学会发现当 package-lock.json 指定的版本号不在 package.json 指定的范围内时,package-lock.json 就会被更新覆盖。这可不利于我们去维持版本的固定。

    因此后续 npm 也推出了 npm ci 的指令来解决这一问题,npm ci 和 npm i 的不同之处在于:当 package-lock.json 指定的依赖版本不在 package.json 指定的依赖版本范围内时,npm 会报错并取消安装。

    这样我们就不怕在 package-lock 和 package.json 不一致时发生覆盖更新。

    总结

    在 npm5.6 以后我们就可以放心大胆地使用 package-lock.json 文件来锁版本,而在构建部署时可以使用 npm ci 安装命令来防止 npm install 的覆盖更新问题。

    写在最后

    我个人开了一个公众号“前端搬运小工”,我会定期推送优秀的前端精选文章,拒绝无脑基础入门的文章,带给你不一样的前端视角。 在这里插入图片描述

    1 条回复    2019-06-21 02:16:23 +08:00
    ochatokori
        1
    ochatokori  
       2019-06-21 02:16:23 +08:00 via Android
    感觉我没碰到过依赖版本的变化带来的问题…

    杠一下,用得最多的是 npm run dev
    关于     帮助文档     自助推广系统     博客     API     FAQ     Solana     3197 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 24ms UTC 10:49 PVG 18:49 LAX 03:49 JFK 06:49
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86