在前端中,什么是幽灵依赖?

前言

大家好,我是林三心,用最通俗易懂的话讲最难的知识点是我的座右铭,基础是进阶的前提是我的初心。

幽灵依赖?

前几天在公司的技术分享会上,我总是听到大佬们在提起一个名词——幽灵依赖,起初我没有太在意,以为这个不太重要,所以就没怎么去了解这个名词。

直到我在做项目pnpm迁移的时候,我突然很想知道它跟幽灵依赖到底有什么关系呢?

其实在了解什么关系之前,我应该去了解一下什么是幽灵依赖,我们接着聊!!!

什么是幽灵依赖?

比如我现在,我在开发一个项目A,项目A中我装了npm-lsx这个包,而这个npm-lsx的包依赖了npm-test,咱们来看看这两处packages.json

依赖关系是:A <- npm-lsx <- npm-test

// A/packages.json

{
  "name""A",
  "dependencies": {
    "npm-lsx""^1.0.0"
  }
}

// A/node_modules/npm-lsx/packages.json

{
  "name""npm-lsx",
  "dependencies": {
    "npm-test""^1.0.0"
  }
}

node_modules规则

如果按照node_modules的规则的话,那么目录的结构应该是这样的

node_modules
├── npm-lsx 
|   └── node_modules
|       └── npm-test

但是如果依赖关系很长的话,那么会导致目录深度非常深,所以我们常用的npm、yarn解决了这个问题

npm、yarn

它们是怎么解决这个问题的呢?npm、yarn为了解决依赖关系过长时,导致的目录结构过深,采用了扁平化,也就是所有依赖都被拍平到node_modules目录下。这样的好处就是,不再有层级过深的问题。

现在目录结构变成这样了

// A/node_modules

node_modules
├── npm-lsx 
├── npm-test

幽灵依赖

接着上面的示例,继续聊,请看下面的代码

const lsx = require('npm-lsx')
const test = require('npm-test')

lsx()
test()

你们觉得这段代码有问题吗?其实运行起来是没问题的。但是问题来了,我们项目中居然能直接引用npm-test这个包!!!

我们都知道依赖关系是:

A <- npm-lsx <- npm-test

按理说,A是不能直接引用npm-test的,因为没有直接依赖关系啊!!!但是因为前面说了,npm、yarn会将依赖拍平在A的node_modules中,这导致了A可以直接require('npm-test')

我们称这个npm-test幽灵依赖!!!

幽灵依赖的坏处?

某天 npm-lsx 不依赖 npm-test 了

已知你现在代码是这样

const lsx = require('npm-lsx')
const test = require('npm-test')

lsx()
test()

某天npm-lsx升级了!它不再依赖npm-test了!那么此时Anode-modules中变了!

// 以前
node_modules
├── npm-lsx 
├── npm-test

// 现在
node_modules
├── npm-lsx

那么你的代码会报错!

const lsx = require('npm-lsx')
const test = require('npm-test'// 没有

lsx()
test() // 报错!!!

其实这个情况还好,因为这种情况在打包上线过程中就会报错依赖找不到了,所以不太会造成线上的报错崩溃

多项目引用同一个幽灵依赖

我现在有两个项目AB

A中的node_modules目录为,现阶段npm-test的版本为1.0.0

// A/node_modules

node_modules
├── npm-lsx 
├── npm-test 版本号:1.0.0

A中某个文件的代码为

const test = require('npm-test')

test()

B中某个文件的代码为

const test = require('npm-test')

test()

当有一天npm-lsx所依赖的npm-test升级了!版本升级为了2.0.0

// A/node_modules

node_modules
├── npm-lsx 
├── npm-test 版本号:2.0.0

此时test的用法也变了~而因为A项目已经回归过了,所以他知道,也改了对应的代码

const test = require('npm-test')

test.run() // 修改代码

B项目就没那么好运了,它并没有进行回归,所以并没有去改代码!!!!

const test = require('npm-test'// 有

test() // 直接报错

这就惨了,B项目在打包阶段并不会因为依赖不到npm-test而报错,所以它会顺利上线。。然后,等到了线上运行起来,直接报错!!!这是非常严重的的事故!

结语

点个【赞】和【在看】是对林三心最大的鼓励,林三心会非常开心的~~~

关注公众号【前端之神】,回复【加群】,即可获得加入【千人前端学习大群】的方式,交流找工作,摸鱼,吹水~

相关推荐

  • 说点扎心的真相吧...
  • 开源模型打败GPT-4!
  • 【深度学习】一文打尽目标检测NMS | 精度提升篇
  • 【深度学习】使用 YOLOv9+SAM 进行动态物体检测和分割
  • 学生早操死亡,辅导员被撤销教师资格
  • 张俊林:我对LLAMA-3的一点看法!
  • [开源] 好用的 uniapp 跨端解决方案,命令行运行 Web、小程序 和 APP
  • 介绍几位新认识了大佬!
  • 数据竞赛三剑客变四剑客了~
  • 如何掌管世界最大的主权基金:挪威主权基金CEO尼古拉·唐根长篇访谈 · 全文+视频
  • 一手体验Rokid新品!苹果空间计算的软肋,被这家中国AR龙头打成王牌
  • 再见,AI意识先驱:Daniel Dennett
  • 小冰徐元春:AIGC已经让普通人开始赚钱 | 中国AIGC产业峰会
  • 离你最近的空间计算打开方式,X499元?
  • 也看跨模态大模型文档图表理解的数据工程:UniChar、MATCHA等代表模型的数据构造方案
  • 恭喜 BLG!值得!!
  • 明明只学会了 Hello World,却以为掌握了整个 World!
  • 5.5K Star真不错!简化数据收集,轻松创建交互式表单
  • Hadoop1X,Hadoop2X和hadoop3X有很大的区别么?
  • Nest 实现 GraphQL 版 TodoList