vue父子组件通信(二)祖先调用inject

inject 是 Vue 中的一个 依赖注入(Dependency Injection)机制,用于:

跨层级组件传值(祖先组件 → 后代组件)

它通常和 provide 配对使用。

作用是:

避免一层一层通过 props 往下传(prop drilling)。


1. 为什么需要 inject?

正常父子通信:

复制代码
复制代码
App
 └── Parent
      └── Child
           └── GrandChild

如果 App 的数据要给 GrandChild

通常要:

复制代码
复制代码
App → Parent → Child → GrandChild

一级一级:

复制代码
复制代码
<Parent :theme="theme" />

再:

复制代码
复制代码
<Child :theme="theme" />

再:

复制代码
复制代码
<GrandChild :theme="theme" />

很麻烦。

于是 Vue 提供:

provide / inject

让后代组件直接拿。


2. 基本用法

祖先组件 provide

复制代码
复制代码
<script>
export default {
  provide() {
    return {
      theme: 'dark'
    }
  }
}
</script>

后代组件 inject

复制代码
复制代码
<script>
export default {
  inject: ['theme'],

  created() {
    console.log(this.theme)
  }
}
</script>

输出:

复制代码
复制代码
dark

即使中间隔了很多层组件,也能拿到。


3. 更完整例子

App.vue

复制代码
复制代码
<script>
export default {
  provide() {
    return {
      username: 'Tom'
    }
  }
}
</script>

<template>
  <Parent />
</template>

Parent.vue

复制代码
复制代码
<template>
  <Child />
</template>

不需要传 props。


Child.vue

复制代码
复制代码
<script>
export default {
  inject: ['username']
}
</script>

<template>
  <div>{{ username }}</div>
</template>

页面:

复制代码
复制代码
Tom

4. inject 和 props 的区别

对比 props inject
通信方式 父 → 子 祖先 → 任意后代
是否逐层传递 不需要
是否明确 明确 隐式
推荐场景 普通组件通信 全局/跨层级共享

5. 常见场景

① UI 组件库

比如:

复制代码
复制代码
<Form>
   <FormItem>
      <Input />
   </FormItem>
</Form>

Input 要知道:

  • label 宽度
  • 校验状态
  • form 配置

难道一级级 props?

不会。

通常:

复制代码
复制代码
provide('formContext')
inject('formContext')

Element Plus、Ant Design Vue 都大量使用。


② 全局配置

例如:

主题:

复制代码
复制代码
theme
locale
size
permission

③ 插件开发

Vue Router、Pinia、i18n 底层也用了类似机制。

比如:

复制代码
复制代码
app.provide(routerKey, router)

然后:

复制代码
复制代码
inject(routerKey)

拿 router 实例。


6. 响应式问题(重要)

很多人踩坑。

错误理解

以为:

复制代码
复制代码
provide() {
  return {
    count: 0
  }
}

改了会响应更新。

不一定。

因为普通值不是响应式共享。


正确方式(Vue3)

ref/reactive

复制代码
复制代码
import { ref } from 'vue'

const count = ref(0)

provide('count', count)

子组件:

复制代码
复制代码
const count = inject('count')

直接响应式更新。


7. Vue2 写法

复制代码
复制代码
provide() {
  return {
    user: this.user
  }
}

但:

复制代码
复制代码
this.user = 'new'

可能不更新。

Vue2 对 provide 响应式支持比较弱。

通常要:

复制代码
复制代码
provide() {
  return {
    state: this
  }
}

然后:

复制代码
复制代码
inject: ['state']

访问:

复制代码
复制代码
state.user

8. 一句话理解

props 是:

父组件显式传参

inject 是:

后代组件向祖先"拿依赖"

有点像 Java/Spring:

复制代码
复制代码
@Autowired
UserService userService;

祖先:

复制代码
复制代码
@Bean

后代:

复制代码
复制代码
@Inject

所以很多人把 Vue 的 provide/inject 理解成:

轻量级依赖注入(DI)机制

相关推荐
To_OC3 小时前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC4 小时前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
天渺工作室4 小时前
实现一个adblock/adblock plus等浏览器广告拦截器检测插件
前端·javascript
阳光是sunny5 小时前
Vue 项目怎么做用户行为全链路监控?轻量插件方案详解
前端·面试·架构
ZhengEnCi5 小时前
Q04-Vite禁用CSS代码分割-解决生产环境样式加载顺序混乱问题
前端·vue.js·vite
九酒5 小时前
AI Agent 开发踩坑记:口播功能非得用 APP 原生实现吗?
前端·人工智能·agent
Jackson__6 小时前
做了一段时间的AI coding后,我终于搞清了 CLI 和 MCP 的区别
前端·agent·ai编程
IT_陈寒8 小时前
JavaScript项目实战经验分享
前端·人工智能·后端
用户47949283569159 小时前
6w star,GitHub 趋势第一的 Ponytail,这个agent插件到底在火什么
前端·后端
薛定喵的谔10 小时前
我开源了一个精致的 Next.js 博客模板:Skyplume
前端·前端框架·next.js