当前位置: 首页 > news >正文

Vue3 provide/inject用法总结

1. 基本概念

provide/inject 是 Vue3 中实现跨层级组件通信的方案,类似于 React 的 Context。它允许父组件向其所有子孙组件注入依赖,无论层级有多深。

1.1 基本语法

// 提供方(父组件)
const value = ref('hello')
provide('key', value)// 注入方(子孙组件)
const value = inject('key')

2. 基础用法

2.1 提供静态值

<!-- Parent.vue -->
<script setup>
import { provide } from 'vue'// 提供静态值
provide('theme', 'dark')
provide('language', 'zh-CN')
</script><!-- Child.vue -->
<script setup>
import { inject } from 'vue'// 注入值
const theme = inject('theme')
const language = inject('language')
</script><template><div :class="theme"><p>Current Language: {{ language }}</p></div>
</template>

2.2 提供响应式数据

<!-- Parent.vue -->
<script setup>
import { provide, ref } from 'vue'const count = ref(0)
const updateCount = () => {count.value++
}// 提供响应式数据和更新方法
provide('count', {count,updateCount
})
</script><!-- Child.vue -->
<script setup>
import { inject } from 'vue'const { count, updateCount } = inject('count')
</script><template><div><p>Count: {{ count }}</p><button @click="updateCount">Increment</button></div>
</template>

3. 进阶用法

3.1 使用 Symbol 作为 key

// injection-keys.ts
export const COUNT_KEY = Symbol('count')
export const THEME_KEY = Symbol('theme')
<!-- Parent.vue -->
<script setup lang="ts">
import { provide } from 'vue'
import { COUNT_KEY, THEME_KEY } from './injection-keys'const count = ref(0)
provide(COUNT_KEY, count)
provide(THEME_KEY, 'dark')
</script><!-- Child.vue -->
<script setup lang="ts">
import { inject } from 'vue'
import { COUNT_KEY, THEME_KEY } from './injection-keys'const count = inject(COUNT_KEY)
const theme = inject(THEME_KEY)
</script>

3.2 提供默认值

<script setup>
import { inject } from 'vue'// 使用静态默认值
const theme = inject('theme', 'light')// 使用工厂函数作为默认值
const now = inject('timestamp', () => Date.now())
</script>

3.3 只读数据

<!-- Parent.vue -->
<script setup>
import { provide, ref, readonly } from 'vue'const count = ref(0)
// 提供只读版本,防止子组件修改
provide('count', readonly(count))// 提供更新方法
provide('updateCount', () => {count.value++
})
</script><!-- Child.vue -->
<script setup>
import { inject } from 'vue'const count = inject('count')
const updateCount = inject('updateCount')
</script>

4. 实际应用场景

4.1 主题系统

<!-- ThemeProvider.vue -->
<script setup>
import { provide, ref } from 'vue'const theme = ref('light')
const toggleTheme = () => {theme.value = theme.value === 'light' ? 'dark' : 'light'
}provide('theme', {theme,toggleTheme
})
</script><template><div :class="theme.value"><slot></slot></div>
</template><!-- 使用组件 -->
<script setup>
import { inject } from 'vue'const { theme, toggleTheme } = inject('theme')
</script><template><button @click="toggleTheme">Switch to {{ theme === 'light' ? 'dark' : 'light' }} mode</button>
</template>

4.2 多语言系统

<!-- I18nProvider.vue -->
<script setup>
import { provide, ref } from 'vue'const locale = ref('en')
const messages = {en: {greeting: 'Hello',farewell: 'Goodbye'},zh: {greeting: '你好',farewell: '再见'}
}const t = (key) => messages[locale.value][key]
const setLocale = (lang) => {locale.value = lang
}provide('i18n', {locale,t,setLocale
})
</script><!-- 使用组件 -->
<script setup>
import { inject } from 'vue'const { t, setLocale, locale } = inject('i18n')
</script><template><div><select v-model="locale"><option value="en">English</option><option value="zh">中文</option></select><p>{{ t('greeting') }}</p></div>
</template>

4.3 状态管理

<!-- Store.vue -->
<script setup>
import { provide, reactive } from 'vue'const store = reactive({user: null,todos: [],addTodo(text) {this.todos.push({ id: Date.now(), text, completed: false })},toggleTodo(id) {const todo = this.todos.find(t => t.id === id)if (todo) {todo.completed = !todo.completed}}
})provide('store', store)
</script><!-- TodoList.vue -->
<script setup>
import { inject } from 'vue'const store = inject('store')
</script><template><div><inputv-model="newTodo"@keyup.enter="store.addTodo(newTodo)"><ul><liv-for="todo in store.todos":key="todo.id"@click="store.toggleTodo(todo.id)">{{ todo.text }}</li></ul></div>
</template>

5. TypeScript 支持

5.1 类型定义

// types.ts
export interface ThemeContext {theme: Ref<'light' | 'dark'>toggleTheme: () => void
}export const ThemeSymbol = Symbol('theme')

5.2 带类型的 provide/inject

<script setup lang="ts">
import { provide, inject } from 'vue'
import { ThemeContext, ThemeSymbol } from './types'// 提供方
provide<ThemeContext>(ThemeSymbol, {theme: ref('light'),toggleTheme: () => { /* ... */ }
})// 注入方
const theme = inject<ThemeContext>(ThemeSymbol)
</script>

6. 最佳实践

  1. 使用 Symbol 作为 key

    const MyKey = Symbol('my-key')
    provide(MyKey, value)
    
  2. 提供只读数据

    provide('data', readonly(data))
    
  3. 集中管理 injection key

    // keys.ts
    export const THEME_KEY = Symbol('theme')
    export const I18N_KEY = Symbol('i18n')
    
  4. 使用组合式函数封装

    // useTheme.ts
    export function useTheme() {const theme = inject(THEME_KEY)if (!theme) {throw new Error('useTheme must be used within ThemeProvider')}return theme
    }
    

7. 注意事项

  1. 响应性

    • 确保提供响应式数据时使用 ref 或 reactive
    • 注意数据的可变性和只读性
  2. 默认值

    • 提供合理的默认值
    • 考虑使用工厂函数作为默认值
  3. 类型安全

    • 使用 TypeScript 定义接口
    • 使用 Symbol 作为 key
  4. 性能考虑

    • 避免提供过大的数据结构
    • 合理划分提供的数据范围

通过合理使用 provide/inject,我们可以有效地管理跨组件通信,构建可维护的组件树。但要注意避免过度使用,以免造成数据流向难以追踪的问题。

相关文章:

Vue3 provide/inject用法总结

1. 基本概念 provide/inject 是 Vue3 中实现跨层级组件通信的方案&#xff0c;类似于 React 的 Context。它允许父组件向其所有子孙组件注入依赖&#xff0c;无论层级有多深。 1.1 基本语法 // 提供方&#xff08;父组件&#xff09; const value ref(hello) provide(key, …...

Linux——网络基础(1)

文章目录 目录 文章目录 前言 一、文件传输协议 应用层 传输层 网络层 数据链路层 数据接收与解封装 主机与网卡 数据传输过程示意 二、IP和MAC地址 定义与性质 地址格式 分配方式 作用范围 可见性与可获取性 生活例子 定义 用途 特点 联系 四、TCP和UDP协…...

【记录】日常|从零散记录到博客之星Top300的成长之路

文章目录 shandianchengzi 2024 年度盘点概述写作风格简介2024年的创作内容总结 shandianchengzi 2024 年度盘点 概述 2024年及2025年至今我创作了786即84篇文章&#xff0c;加上这篇就是85篇。 很荣幸这次居然能够入选博客之星Top300&#xff0c;这个排名在我之前的所有年份…...

【二分查找】力扣373. 查找和最小的 K 对数字

给定两个以 非递减顺序排列 的整数数组 nums1 和 nums2 , 以及一个整数 k 。 定义一对值 (u,v)&#xff0c;其中第一个元素来自 nums1&#xff0c;第二个元素来自 nums2 。 请找到和最小的 k 个数对 (u1,v1), (u2,v2) … (uk,vk) 。 示例 1: 输入: nums1 [1,7,11], nums2 …...

池化层Pooling Layer

1. 定义 池化是对特征图进行的一种压缩操作&#xff0c;通过在一个小的局部区域内进行汇总统计&#xff0c;用一个值来代表这个区域的特征信息&#xff0c;常用于卷积神经网络&#xff08;CNN&#xff09;中。 2. 作用 提取代表性信息的同时降低特征维度&#xff0c;具有平移…...

力扣算法题——11.盛最多水的容器

目录 &#x1f495;1.题目 &#x1f495;2.解析思路 本题思路总览 借助双指针探索规律 从规律到代码实现的转化 双指针的具体实现 代码整体流程 &#x1f495;3.代码实现 &#x1f495;4.完结 二十七步也能走完逆流河吗 &#x1f495;1.题目 &#x1f495;2.解析思路…...

自由学习记录(32)

文件里找到切换颜色空间 fgui中的 颜色空间是一种总体使用前的设定 颜色空间&#xff0c;和半透明混合产生的效果有差异&#xff0c;这种问题一般可以产生联系 动效就是在fgui里可以编辑好&#xff0c;然后在unity中也准备了对应的调用手段&#xff0c;可以详细的使用每一个具…...

VScode+Latex (Recipe terminated with fatal error: spawn xelatex ENOENT)

使用VSCode编辑出现Recipe terminated with fatal error: spawn xelatex ENOENT问题咋办&#xff1f; 很好解决&#xff0c;大概率的原因是因为latex没有添加到系统环境变量中&#xff0c;所有设置的编译工具没有办法找到才出现的这种情况。 解决方法&#xff1a; winR 然后输…...

「蓝桥杯题解」蜗牛(Java)

题目链接 这道题我感觉状态定义不太好想&#xff0c;需要一定的经验 import java.util.*; /*** 蜗牛* 状态定义&#xff1a;* dp[i][0]:到达(x[i],0)最小时间* dp[i][1]:到达 xi 上方的传送门最小时间*/public class Main {static Scanner in new Scanner(System.in);static f…...

PHP EOF (Heredoc) 详解

PHP EOF (Heredoc) 详解 PHP 中的 EOF(End Of File)是一种非常有用的语法特性,允许开发者创建多行字符串。它特别适合于创建格式化文本,如配置文件、HTML 模板等。本文将详细讲解 PHP EOF 的用法、优势以及注意事项。 什么是 EOF? EOF 是一种特殊的字符串定义方式,它允…...

pyautogui操控Acrobat DC pro万能PDF转Word,不丢任何PDF格式样式

为了将PDF转换脚本改为多进程异步处理&#xff0c;我们需要确保每个进程独立操作不同的Acrobat窗口。以下是实现步骤&#xff1a; 实现代码 import os import pyautogui import time import subprocess import pygetwindow as gw from multiprocessing import Pooldef conver…...

Day32:字符串的复制

在 Python 中&#xff0c;字符串的复制是指创建一个新的字符串&#xff0c;它的内容与原字符串相同。字符串是不可变的对象&#xff0c;这意味着你不能直接修改字符串的内容&#xff0c;但是可以通过复制来创建新的字符串进行操作。字符串的复制在一些情况下非常有用&#xff0…...

基于Mybatis继承AbstractRoutingDataSource使用自定义注解实现动态数据源

一&#xff1a;实现 方式一&#xff1a;继承AbstractRoutingDataSource使用自定义注解实现 环境&#xff1a;springboot3 MyBatis3 mysql-connector8 DataSourceKeyEnum枚举类 有几个数据源就配置几个枚举类&#xff0c;和数据源数量一一对应 class DataSourceKeyEnum{D…...

ZooKeeper 数据模型

ZooKeeper 数据模型 ZooKeeper 拥有层次化的命名空间&#xff0c;类似分布式文件系统&#xff0c;但每个节点不仅能有子节点&#xff0c;还可关联数据。节点路径为规范的绝对路径&#xff0c;用斜杠分隔&#xff0c;无相对引用。路径命名有如下约束&#xff1a; 路径名不能包…...

【VUE】Vue2中Vue.extend方法

在 Vue.js 2.x 版本中&#xff0c;Vue.extend() 方法被用于创建一个新的 Vue 子类&#xff0c;可以在该子类上扩展一些属性、指令和组件选项等&#xff0c;然后进行实例化。 比如&#xff0c;可以在创建一些类似 loading 式的函数式插件时&#xff0c;使用&#xff1a; 在 Vue…...

MaskGAE论文阅读

What’s Behind the Mask: Understanding Masked Graph Modeling for Graph Autoencoders 碎碎念&#xff1a;一篇论文看四天&#xff0c;效率也没谁了(捂脸) 看一点忘一点&#xff0c;虽然在本子上有记录&#xff0c;但还是忘&#xff0c;下次看一点在博客上记一点启发 本来很…...

Mybatis-plus 更新 Null 的策略踩坑记

一个bug 在一个管理页面&#xff0c;有一个非必填字段被设置成空了并提交更新&#xff0c;再次打开的时候&#xff0c;发现字段还在&#xff0c;并没有被更新成功。 使用的数据库映射框架是 Mybatis-plus &#xff0c;对于Mybatis 在更新字段的时候会对空进行校验&#xff0c;…...

Oracle迁移DM数据库

Oracle迁移DM数据库 本文记录使用达梦官方数据迁移工具DTS&#xff0c;将Oracle数据库的数据迁移至达梦数据库。 1 数据准备 2 DTS工具操作步骤 2.1 创建工程 打开DTS迁移工具&#xff0c;点击新建工程&#xff0c;填写好工程信息&#xff0c;如图&#xff1a; 2.2 新建迁…...

HTML特殊符号的使用示例

目录 一、基本特殊符号的使用 1、空格符号&#xff1a; 2、小于号 和 大于号&#xff1a; 3、引号&#xff1a; 二、版权、注册商标符号的使用 1、版权符号&#xff1a;© 2、注册商标符号&#xff1a; 三、数学符号的使用 四、箭头符号的使用 五、货币符号的使用…...

数据结构基础之《(15)—排序算法小结》

一、排序算法的稳定性 1、稳定性是指同样大小的样本再排序之后不会改变相对次序 2、对基础类型来说&#xff0c;稳定性毫无意义 比如&#xff1a;3和3没有区别。《潜伏》里说同样两个一百元大钞&#xff0c;你能告诉我哪一个是高尚的那一个是龌龊的么 3、对非基础类型来说&a…...

苍穹外卖day10(黑马程序员)

苍穹外卖 day10 笔记 WebSocket 什么是 WebSocket WebSocket 是一种全双工的网络通信方式&#xff1a;客户端和服务器建立连接之后&#xff0c;双方都可以随时主动给对方发消息&#xff0c;不必像传统网页那样「每次都要重新发起一次请求」。 可以把它理解成&#xff1a; HTTP&…...

本科论文知网AI率高的原因和解决方法全在这里

知网AIGC检测出来AI率高&#xff0c;很多同学第一反应是"我没有全程用AI写啊&#xff0c;为什么这么高&#xff1f;"这个问题确实需要好好解释一下——知网检测到的AI率高&#xff0c;未必是因为你完全靠AI写的。 知网AIGC检测是怎么工作的 知网的AIGC检测系统会分…...

基于stm32单片机的智能导盲系统的设计与实现

一、摘要 为提高视障人群出行的安全性与便捷性&#xff0c;设计并实现一套基于STM32F103C8T6单片机的智能导盲系统。系统集成多种传感器与功能模块&#xff0c;能够实现环境感知、位置信息反馈、跌倒检测与紧急求助等功能。通过SR04超声波传感器实时监测前方障碍物&#xff0c;…...

SClick技术解析:防休眠工具的工作原理探讨

SClick是一款轻量级的防休眠工具&#xff0c;能够帮助用户解决Windows系统自动休眠带来的诸多不便。 软件体积仅有几十KB&#xff0c;绿色便携&#xff0c;无需安装&#xff0c;即用即走。 它通过模拟鼠标点击的方式&#xff0c;让系统以为用户一直在操作电脑&#xff0c;从而防…...

AI辅助开发:让快马AI理解并生成ccswitch工具的核心逻辑与UI管理代码

AI辅助开发&#xff1a;让快马AI理解并生成ccswitch工具的核心逻辑与UI管理代码 最近在开发一个网络切换工具ccswitch时&#xff0c;发现AI辅助开发能大幅提升效率。通过InsCode(快马)平台集成的AI模型&#xff0c;可以用自然语言描述需求&#xff0c;就能自动生成核心功能代码…...

Alexa Plus 拓展食品配送领域,语音订餐体验升级

Alexa Plus 开启食品配送新功能从本周起&#xff0c;Alexa Plus 拓展至食品配送领域&#xff0c;用户可通过它从优步外卖&#xff08;Uber Eats&#xff09;和 Grubhub 订餐。只需将优步或 Grubhub 应用与 Alexa Plus 设备关联&#xff0c;就能询问食品配送情况&#xff0c;并通…...

安规标准考核题库-2(IEC 62477-1:2012+AMD1:2022 )

本题库严格对标 IEC 62477-1:2012《电力电子变换器系统和设备安全要求 第 1 部分&#xff1a;总则》AMD1:2022 修订版&#xff0c;贴合储能双向 PCS 的设计、测试、认证全流程场景&#xff0c;分为判断题、单选题、多选题三类&#xff0c;所有题目均附标准条款依据与详细解答。…...

解密Abaqus许可证“心跳”机制与合理超时时间设置

解密Abaqus许可证“心跳”机制跟合理超时时间设置你是不单是也碰到过这种情况&#xff1a;Abaqus许可证明明用不了&#xff0c;可系统还在继续计费&#xff1f;我在一家制造企业做许可证优化&#xff0c;就碰到了此老问题。为何许可证会“死掉”&#xff1f;这跟许可证的心跳&a…...

百考通:AI精准赋能开题报告,让学术研究全流程智能化

对于每一位学子与科研人而言&#xff0c;开题报告是学术研究的“第一粒扣子”&#xff0c;它不仅是研究方向的蓝图&#xff0c;更是顺利推进论文写作、获得导师认可的关键。然而&#xff0c;选题迷茫、文献梳理繁琐、逻辑框架搭建困难等问题&#xff0c;常常让开题之路步履维艰…...

高新申报通关指南:资深工程师手把手教你准备全套材料(附清单)

高新技术企业认定不仅是企业税负减免的“金钥匙”&#xff0c;更是研发实力与技术体系的重要证明。很多技术人员觉得这只是财务或行政的事&#xff0c;但实则技术材料的质量直接决定了申报成败。 一、 基础资质与人员结构 这部分是申报的“地基”&#xff0c;务必确保真实且符合…...