npm的lock機(jī)制解析
npm是什么
npm是一個(gè)包管理工具,開源作者可以把開源包發(fā)布在平臺(tái)上供其他人下載使用。前端的同學(xué)基本都使用過npm,這里就不做過多介紹。日常工作中npm的主要用途就是根據(jù)項(xiàng)目的package.json使用npm install去安裝依賴。
npm install可以說是我們使用最頻繁的一個(gè)指令。在npm5版本之前,npm install會(huì)根據(jù)package.json指定的依賴版本去進(jìn)行安裝。但往往package.json中指定的是一個(gè)版本范圍,例如:
"dependencies": {
"packageA": "^2.0.0"
},
以上這個(gè) ^2.0.0 指定的范圍是版本號(hào)大于等于2.0.0且大版本號(hào)為2。即2.6.10這個(gè)是符合的,而3.0.0和1.0.0這種是不符合的。
這樣的范圍指定會(huì)導(dǎo)致一個(gè)問題:A新建了一個(gè)項(xiàng)目,生成了上面這份package.json文件,但A安裝依賴的時(shí)間比較早,此時(shí)packageA的最新版本是2.1.0,該版本與代碼兼容,沒有出現(xiàn)bug。后來B克隆了A的項(xiàng)目,在安裝依賴時(shí)packageA的最新版本是2.2.0,那么根據(jù)語義npm會(huì)去安裝2.2.0的版本,但2.2.0版本的API可能發(fā)生了改動(dòng),導(dǎo)致代碼出現(xiàn)bug。
這就是package.json會(huì)帶來的問題,同一份package.json在不同的時(shí)間和環(huán)境下安裝會(huì)產(chǎn)生不同的結(jié)果。
理論上這個(gè)問題是不應(yīng)該出現(xiàn)的,因?yàn)閚pm作為開源世界的一部分,也遵循一個(gè)發(fā)布原則:相同大版本號(hào)下的新版本應(yīng)該兼容舊版本。即2.1.0升級(jí)到2.2.0時(shí)API不應(yīng)該發(fā)生變化。
但很多開源庫(kù)的開發(fā)者并沒有嚴(yán)格遵守這個(gè)發(fā)布原則,導(dǎo)致了上面的這個(gè)問題。
lock機(jī)制
一個(gè)新的事物的誕生都是為了解決一個(gè)歷史問題
基于這種狀況,npm5推出了lock機(jī)制。在使用npm5.0.0之后的版本時(shí),npm install后會(huì)自動(dòng)生成package-lock.json文件,該文件記錄了當(dāng)前這次install所安裝的依賴版本號(hào)。
例如當(dāng)package.json的依賴如下:
"dependencies": {
"vue": "^2.0.0"
},
install后自動(dòng)生成的package-lock.json會(huì)指定安裝vue2.6.10版本(當(dāng)前最新)
"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相當(dāng)于本次install的一個(gè)快照,它不僅記錄了package.json指明的直接依賴的版本,也記錄了間接依賴的版本。
如果我們想在不同環(huán)境和不同時(shí)間下每次install時(shí)安裝相同版本的依賴,我們就可以把package-lock.json帶上。
當(dāng)package.json和package-lock.json同時(shí)存在時(shí),npm install會(huì)去檢測(cè)package-lock.json指定的依賴版本是否在package.json指定的范圍內(nèi)。如果在,則安裝package-lock.json指定的版本。如果不在,則忽略package-lock.json,并且用安裝的新版本號(hào)覆蓋package-lock.json。
舉個(gè)例子:
// 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指定的范圍內(nèi),npm install會(huì)安裝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指定的范圍內(nèi),npm install會(huì)按照^2.2.0的規(guī)則去安裝最新的2.6.10版本,并且將package-lock.json的版本更新為2.6.10。
值得注意的是npm5一發(fā)布時(shí)并不是采取這種install邏輯,在npm5.0到npm5.6之間install的邏輯發(fā)生了多次變更,而在npm5.6之后一直沿用當(dāng)前這種邏輯。
npm ci
npm5之后的lock機(jī)制滿足了要求鎖版本的開發(fā)者們的需要,我們只需要拿到一份package-lock.json就可以知道要安裝的依賴的具體版本號(hào)。但細(xì)心的同學(xué)會(huì)發(fā)現(xiàn)當(dāng)package-lock.json指定的版本號(hào)不在package.json指定的范圍內(nèi)時(shí),package-lock.json就會(huì)被更新覆蓋。這可不利于我們?nèi)ゾS持版本的固定。
因此后續(xù)npm也推出了npm ci的指令來解決這一問題,npm ci和npm i的不同之處在于:當(dāng)package-lock.json指定的依賴版本不在package.json指定的依賴版本范圍內(nèi)時(shí),npm會(huì)報(bào)錯(cuò)并取消安裝。

這樣我們就不怕在package-lock和package.json不一致時(shí)發(fā)生覆蓋更新。
總結(jié)
在npm5.6以后我們就可以放心大膽地使用package-lock.json文件來鎖版本,而在構(gòu)建部署時(shí)可以使用npm ci安裝命令來防止npm install的覆蓋更新問題。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
利用node實(shí)現(xiàn)一個(gè)批量重命名文件的函數(shù)
這篇文章主要給大家介紹了關(guān)于利用node實(shí)現(xiàn)一個(gè)批量重命名文件的函數(shù)的相關(guān)資料,文中通過示例示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-12-12
npm安裝依賴報(bào)錯(cuò)ERESOLVE?unable?to?resolve?dependency?tree的解決方
當(dāng)我們拿到一個(gè)前端項(xiàng)目的時(shí)候,想要把它運(yùn)行起來,首先是要給它安裝依賴,下面這篇文章主要給大家介紹了關(guān)于npm安裝依賴報(bào)錯(cuò)ERESOLVE?unable?to?resolve?dependency?tree的解決方法,需要的朋友可以參考下2023-04-04
Node.js腳本提取OPML文件信息實(shí)現(xiàn)示例詳解
這篇文章主要為大家介紹了Node.js腳本提取OPML文件信息,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
node-sass安裝失敗解決方法總結(jié)(終有一款適合你)
node-sass是我們開發(fā)中很常見的依賴包,也是安裝時(shí)間冗長(zhǎng)和最常見到報(bào)錯(cuò)的依賴,下面這篇文章主要給大家介紹了關(guān)于node-sass安裝失敗解決方法的相關(guān)資料,需要的朋友可以參考下2022-11-11
Sublime Text3 配置 NodeJs 環(huán)境的方法
大家都知道,Sublime Text 安裝插件一般從 Package Control 中直接安裝即可,當(dāng)我安裝 node js 插件時(shí)候,直接通過Package Control 安裝,雖然插件安裝成功了,但是找不到配置文件 Nodejs.sublime-build 來更改一些配置2020-05-05

