浅析vue中的provide / inject 有什么用处


Posted in Javascript onNovember 10, 2019

1.前言

vue的父子组件通信用什么?

:prop和$emit的组合。

如果是爷孙组件呢?

:那么就要用父组件来转发数据和事件了。

如果是太爷爷和孙子组件呢?

:当然是vuex啦

emmm 好的,没我啥事了,我这就走。

不行,我还能再挣扎一会儿!肯定有一部分兄弟做的项目比较小,组件通信的情况不是很多,懒得引入vuex,那么provide/inject就是爷孙(不限于爷孙/父子,中间隔了多少级都可以)通信问题的最好解决方案啦!

2.官方文档抄过来的介绍

这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。

provide 选项应该是

  • 一个对象或返回一个对象的函数。该对象包含可注入其子孙的属性。在该对象中你可以使用 ES2015 Symbols 作为 key,但是只在原生支持 Symbol 和 Reflect.ownKeys 的环境下可工作。

inject 选项应该是:

  • 一个字符串数组,或
  • 一个对象(详情点击 这里 )

3.基本用法

// 祖先组件 提供foo
//第一种
export default {
 name: "grandfather",
 provide(){
  return{
   foo:'halo'
  }
 },
}
//第二种
export default {
 name: "grandfather",
 provide:{
  foo:'halo~~~~'
 },
}
//后代组件 注入foo
export default {
 inject:['foo'],
}

上面的两种用法有什么区别吗?

  • 如果你只是传一个字符串,像上面的‘halo',那么是没有区别的,后代都能读到。
  • 如果你需要传一个对象(如下所示代码),那么第二种是传不了的,后代组件拿不到数据。所以建议只写第一种
//当你传递对象给后代时
provide(){
  return{
   test:this.msg
  }
 },

注意: 一旦注入了某个数据,比如上面示例中的 foo,那这个组件中就不能再声明 foo 这个数据了,因为它已经被父级占有。

4.什么时候才是可响应的?

如果你使用过vuex,那么你会认为,上面的例子中,如果我把 foo:'halo' 改成 foo:'world' ,子组件会及时响应数据变更,视图完成更新。

可惜,没有

在vue官方文档中有这么一句话

提示: provide 和 inject 绑定并不是可响应的。这是刻意为之的。然而,如果你传入了一个可监听的对象,那么其对象的属性还是可响应的。

这里不探讨vue为什么要这么设计,我这里只展示啥时候provide/inject可响应

provide(){
 return{
  test:this.msg
 }
},
data() {
 return {
  msg: "Welcome to Your Vue.js App",
 }
}
mounted(){
 setTimeout(()=>{
  this.msg = "halo world";
  console.log(this._provided.msg)
  //log:Welcome to Your Vue.js App
 },3000)
},

如上所示,这样做是不行的,打印出来的 provided 中的数据并没有改,子组件取得值也没变。

你甚至可以直接给 this._provided.msg 赋值,但是 即使 是 _provided.msg 里面的值改变了,子组件的取值,依然没有变。

当你像下面这样做的时候,就可以响应了

provide(){
 return{
  test:this.activeData
 }
},
data() {
 return {
  activeData:{name:'halo'},
 }
}
mounted(){
 setTimeout(()=>{
  this.activeData.name = 'world';
 },3000)
}

这就是vue中写道的 对象的属性 是可以响应的

然而,如果你传入了一个可监听的对象,那么其对象的属性还是可响应的。

5.实现全局变量

全局变量?provide/inject不是只能从祖先传递给后代吗?没错,我们只要绑定到最最顶层的组件即可。

就是你啦! app.vue

我们把一整个实例都扔给后代!

//app.vue
export default {
 name: 'App',
 provide(){
  return{
   app:this
  }
 },
 data(){
  return{
   text:"it's hard to tell the night time from the day"
  }
 },
 methods:{
  say(){
   console.log("Desperado, why don't you come to your senses?")
  }
 }
}
//其他所有子组件,需要全局变量的,只需要按需注入app即可
export default {
 inject:['foo','app'],
 mounted(){
  console.log(this.app.text);//获取app中的变量
  this.app.say();//可以执行app中的方法,变身为全局方法!
 }
}

这个也是可响应的噢~

6.实现页面刷新

1 . 用vue-router重新路由到当前页面,页面是不进行刷新的

2 . 采用window.reload(),或者router.go(0)刷新时,整个浏览器进行了重新加载,闪烁,体验不好

那我们怎么做呢?

跟上面的原理差不多,我们只在控制路由的组件中写一个函数(使用 v-if 控制 router-view 的显示隐藏,这里的原理不作赘述),然后把这个函数传递给后代,然后在后代组件中调用这个方法即可刷新路由啦。

//app.vue
<router-view v-if="isShowRouter"/>
export default {
 name: 'App',
 provide(){
  return{
   reload:this.reload
  }
 },
 data(){
  return{
   isShowRouter:true,
  }
 },
 methods:{
  reload(){
   this.isShowRouter = false;
   this.$nextTick(()=>{
    this.isShowRouter = true;
   })
  }
 }
}
//后代组件
export default {
 inject:['reload'], 
}

7.结尾

vue中有这样的提示

注意: provide 和 inject 主要为高阶插件/组件库提供用例。并不推荐直接用于应用程序代码中。

provide/inject平时用的比较少,多数用于开发组件,但某些情况下还是很好用的。

业务庞大而复杂的,还是建议使用 vuex ~

总结

以上所述是小编给大家介绍的vue中的provide / inject 有什么用处,希望对大家有所帮助!

Javascript 相关文章推荐
JavaScript脚本性能的优化方法
Feb 02 Javascript
IE8 中使用加速器(Activities)
May 14 Javascript
Bootstrap中CSS的使用方法
Feb 17 Javascript
js实现不重复导入的方法
Mar 02 Javascript
AngularJS通过$sce输出html的方法
Sep 22 Javascript
js输入框使用正则表达式校验输入内容的实例
Feb 12 Javascript
div中文字内容溢出常见的解决方法
Mar 16 Javascript
angular4实现tab栏切换的方法示例
Oct 21 Javascript
vue项目中使用axios上传图片等文件操作
Nov 02 Javascript
js+canvas实现验证码功能
Sep 21 Javascript
es6中class类静态方法,静态属性,实例属性,实例方法的理解与应用分析
Feb 15 Javascript
JavaScript前端面试组合函数
Jun 21 Javascript
Vue项目中使用jsonp抓取跨域数据的方法
Nov 10 #Javascript
webpack4 optimization使用总结
Nov 10 #Javascript
vue ssr服务端渲染(小白解惑)
Nov 10 #Javascript
node后端服务保活的实现
Nov 10 #Javascript
vue动态循环出的多个select出现过的变为disabled(实例代码)
Nov 10 #Javascript
vue父子组件的通信方法(实例详解)
Nov 10 #Javascript
分享Angular http interceptors 拦截器使用(推荐)
Nov 10 #Javascript
You might like
PHP无敌近乎加密方式!
2010/07/17 PHP
codeigniter教程之上传视频并使用ffmpeg转flv示例
2014/02/13 PHP
ecshop后台编辑器替换成ueditor编辑器
2015/03/03 PHP
php计算多维数组中所有值总和的方法
2015/06/24 PHP
详解thinkphp中的volist标签
2018/01/15 PHP
JavaScript新窗口与子窗口传值详解
2014/02/11 Javascript
node.js中的http.response.getHeader方法使用说明
2014/12/14 Javascript
全面解析Bootstrap表单使用方法(表单控件)
2015/11/24 Javascript
微信小程序实现获取自己所处位置的经纬度坐标功能示例
2017/11/30 Javascript
详解用vue2.x版本+adminLTE开源框架搭建后台应用模版
2019/03/15 Javascript
javascript实现的时间格式加8小时功能示例
2019/06/13 Javascript
jQuery zTree插件快速实现目录树
2019/08/16 jQuery
Python的subprocess模块总结
2014/11/07 Python
Python格式化压缩后的JS文件的方法
2015/03/05 Python
python logging 日志轮转文件不删除问题的解决方法
2016/08/02 Python
Python基于递归实现电话号码映射功能示例
2018/04/13 Python
Python和Java的语法对比分析语法简洁上python的确完美胜出
2019/05/10 Python
python os模块在系统管理中的应用
2020/06/22 Python
快速创建python 虚拟环境
2020/11/28 Python
Html5应用程序缓存(Cache manifest)
2018/06/04 HTML / CSS
德国机车企业:FC-Moto
2017/10/27 全球购物
澳大利亚排名第一的狂热牛仔品牌:ONETEASPOON
2018/11/20 全球购物
介绍一下Make? 为什么使用make
2013/12/08 面试题
如何写出好的Java代码
2014/04/25 面试题
Delphi CS笔试题
2014/01/04 面试题
中学生学习生活的自我评价
2013/10/26 职场文书
质量工程师岗位职责
2013/11/16 职场文书
优秀幼教自荐信
2014/02/03 职场文书
大学生军训自我鉴定
2014/02/12 职场文书
护理专科毕业自荐信范文
2014/04/21 职场文书
业务内勤岗位职责
2014/04/30 职场文书
建设单位项目负责人任命书
2014/06/06 职场文书
2014年党支部书记工作总结
2014/12/04 职场文书
三好学生竞选稿
2015/11/21 职场文书
2019新员工试用期转正申请书3篇
2019/08/13 职场文书
详解Python+OpenCV绘制灰度直方图
2022/03/22 Python