Vue 3 学习笔记—Provide 和 Inject 用法及原理

开发 前端
在父子组件传递数据时,通常使用的是 props 和 emit,父传子时,使用的是 props,如果是父组件传孙组件时,就需要先传给子组件,子组件再传给孙组件。

[[439473]]

在父子组件传递数据时,通常使用的是 props 和 emit,父传子时,使用的是 props,如果是父组件传孙组件时,就需要先传给子组件,子组件再传给孙组件,如果多个子组件或多个孙组件使用时,就需要传很多次,会很麻烦。

像这种情况,可以使用 provide 和 inject 解决这种问题,不论组件嵌套多深,父组件都可以为所有子组件或孙组件提供数据,父组件使用 provide 提供数据,子组件或孙组件 inject 注入数据。同时兄弟组件之间传值更方便。

一、Vue2 的 provide / inject 使用

provide :是一个对象,里面是属性和值。如:

  1. provide:{ 
  2.  info:"值" 

 如果 provide 需要使用 data 内的数据时,这样写就会报错。访问组件实例 property 时,需要将 provide 转换为返回对象的函数。

  1. provide(){ 
  2.  return
  3.   info: this.msg 
  4.  } 

 inject :是一个字符串数组。如:

  1. inject: [ 'info' ] 

接收上边 provide 提供的 info 数据,也可以是一个对象,该对象包含 from 和 default 属性,from 是可用做的注入内容中搜索用的 key,default 属性是指定默认值。

在 vue2 中 project / inject 应用:

  1. //父组件 
  2. export default
  3.  provide:{ 
  4.   info:"提供数据" 
  5.  } 
  6. //子组件 
  7. export default
  8.  inject:['info'], 
  9.  mounted(){ 
  10.     console.log("接收数据:", this.info) // 接收数据:提供数据 
  11.  } 
  12. }  

 provide / inject 类似于消息的订阅和发布。provide 提供或发送数据, inject 接收数据。

二、Vue3 的 provide / inject 使用

在组合式 API 中使用 provide/inject,两个只能在 setup 期间调用,使用之前,必须从 vue 显示导入 provide/inject 方法。

provide 函数接收两个参数:

provide( name,value )

name:定义提供 property 的 name 。

value :property 的值。

使用时:

  1. import { provide } from "vue" 
  2. export default { 
  3.   setup(){ 
  4.     provide('info',"值"
  5.   } 

 inject 函数有两个参数:

inject(name,default)

name:接收 provide 提供的属性名。

default:设置默认值,可以不写,是可选参数。

使用时:

  1. import { inject } from "vue" 
  2. export default { 
  3.   setup(){ 
  4.     inject('info',"设置默认值"
  5.   } 

 完整实例1:provide/inject实例

  1. //父组件代码 
  2. <script> 
  3. import { provide } from "vue" 
  4. export default { 
  5.   setup(){ 
  6.     provide('info',"值"
  7.   } 
  8. </script> 
  9.  
  10. //子组件 代码 
  11. <template> 
  12.  {{info}} 
  13. </template> 
  14. <script> 
  15. import { inject } from "vue" 
  16. export default { 
  17.   setup(){ 
  18.     const info = inject('info'
  19.     return
  20.       info 
  21.     } 
  22.   } 
  23. </script>  

三、添加响应性

为了给 provide/inject 添加响应性,使用 ref 或 reactive 。

完整实例2:provide/inject 响应式

  1. //父组件代码 
  2. <template> 
  3.   <div> 
  4.     info:{{info}} 
  5.     <InjectCom ></InjectCom> 
  6.   </div> 
  7. </template> 
  8. <script> 
  9. import InjectCom from "./InjectCom" 
  10. import { provide,readonly,ref } from "vue" 
  11. export default { 
  12.   setup(){ 
  13.     let info = ref("今天你学习了吗?"
  14.     setTimeout(()=>{ 
  15.       info.value = "不找借口,立马学习" 
  16.     },2000) 
  17.     provide('info',info) 
  18.     return
  19.       info 
  20.     } 
  21.   }, 
  22.   components:{ 
  23.     InjectCom 
  24.   } 
  25. </script> 
  26.  
  27. // InjectCom 子组件代码 
  28. <template> 
  29.  {{info}} 
  30. </template> 
  31. <script> 
  32. import { inject } from "vue" 
  33. export default { 
  34.   setup(){ 
  35.     const info = inject('info'
  36.     setTimeout(()=>{ 
  37.       info.value = "更新" 
  38.     },2000) 
  39.     return
  40.       info 
  41.     } 
  42.   } 
  43. </script> 

上述示例,在父组件或子组件都会修改 info 的值。

provide / inject 类似于消息的订阅和发布,遵循 vue 当中的单项数据流,什么意思呢?就是数据在哪,修改只能在哪,不能在数据传递处修改数据,容易造成状态不可预测。

在订阅组件内修改值的时候,可以被正常修改,如果其他组件也使用该值的时候,状态容易造成混乱,所以需要在源头上规避问题。

readonly 只读函数,使用之前需要引入,如果给变量加上 readonly 属性,则该数据只能读取,无法改变,被修改时会发出警告,但不会改变值。

使用方法:

  1. import { readonly } from "vue" 
  2. let info = readonly('只读info值'
  3. setTimout(()=>{ 
  4.  info="更新info" //两秒后更新info的值 
  5. },2000) 

 运行两秒后,浏览器发出警告,提示 info 值不可修改。

所以我们就给provide发射出去的数据,添加一个只读属性,避免发射出去的数据被修改。

完整实例2的 provide 处添加 readonly 。

  1. provide('info', readonly(info)) 

在子组件修改值的时候,会有一个只读提醒。

修改值的时候,还是需要在 provide 发布数据的组件内修改数据,所以会在组件内添加修改方法,同时也发布出去,在子组件处调用就可以了。如:

  1. //发布 
  2. let info = ref("今天你学习了吗?"
  3. const changeInfo = (val)=>{ 
  4.  info.value = val 
  5. provide('info',readonly(info)) 
  6. provide('changeInfo',changeInfo) 
  7.  
  8. //订阅 
  9. const chang = inject('changeInfo'
  10. chang('冲向前端工程师'

 完整示例3:修改数据

  1. // 父组件代码 
  2. <template> 
  3.   <div> 
  4.     info:{{info}} 
  5.     <InjectCom ></InjectCom> 
  6.   </div> 
  7. </template> 
  8.  
  9. <script> 
  10. import InjectCom from "./InjectCom" 
  11. import { provide,readonly,ref } from "vue" 
  12. export default { 
  13.   setup(){ 
  14.     let info = ref("今天你学习了吗?"
  15.     const changeInfo = (val)=>{ 
  16.       info.value = val 
  17.     } 
  18.     provide('info',readonly(info)) 
  19.     provide('changeInfo',changeInfo) 
  20.     return
  21.       info 
  22.     } 
  23.   }, 
  24.   components:{ 
  25.     InjectCom 
  26.   } 
  27. </script> 
  28.  
  29. //InjectCom 子组件代码 
  30. <template> 
  31.   <div> 
  32.     <button @click="chang('冲向前端工程师')">更新值</button> 
  33.   </div> 
  34. </template> 
  35. <script> 
  36. import { inject } from "vue" 
  37. export default { 
  38.   setup(){ 
  39.     const info = inject('info'
  40.     const chang = inject('changeInfo'
  41.     return
  42.       info, 
  43.       chang 
  44.     } 
  45.   } 
  46. </script> 

 

责任编辑:姜华 来源: 今日头条
相关推荐

2021-05-27 10:36:34

ProvideInjectVue3

2021-12-01 08:11:44

Vue3 插件Vue应用

2021-12-07 05:44:45

Vue 3 Watch WatchEffect

2021-12-08 09:09:33

Vue 3 Computed Vue2

2023-11-29 08:49:31

Vue.jsData 函数

2023-11-28 09:03:59

Vue.jsJavaScript

2024-11-11 06:00:00

Vue开发provide

2020-06-09 11:35:30

Vue 3响应式前端

2021-11-16 08:50:29

Vue3 插件Vue应用

2021-11-30 08:19:43

Vue3 插件Vue应用

2021-12-29 07:51:21

Vue3 插件Vue应用

2020-12-01 08:34:31

Vue3组件实践

2021-12-02 05:50:35

Vue3 插件Vue应用

2023-12-11 07:34:37

Computed计算属性Vue3

2024-04-10 08:45:51

Vue 3Proxy对象监测数据

2023-12-14 08:25:14

WatchVue.js监听数据

2021-11-17 08:24:47

Vue3 插件Vue应用

2010-09-14 16:20:19

DIV定位

2010-08-16 15:11:02

DIV

2010-08-26 16:40:35

DIV定位
点赞
收藏

51CTO技术栈公众号