vue.js

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript类库 > vue.js > vue中provide和inject的用法

vue中provide和inject的用法及说明(vue组件爷孙传值)

作者:是小橙鸭丶

这篇文章主要介绍了vue中provide和inject的用法及说明(vue组件爷孙传值),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

provide和inject的用法(vue组件爷孙传值)

聊聊概念

成对出现:provide和inject是成对出现的

作用:用于父组件向子孙组件传递数据

使用方法:provide在父组件中返回要传给下级的数据,inject在需要使用这个数据的子辈组件或者孙辈等下级组件中注入数据。

使用场景:由于vue有$parent属性可以让子组件访问父组件。但孙组件想要访问祖先组件就比较困难。通过provide/inject可以轻松实现跨级访问父组件的数据

使用示例DEMO

父组件:通过provide指定传递给子孙组件的值和方法

<template>
	<div id="app">
		我是父组件:{{message}}
		<second></second>
	</div>
</template>
<script>
import second from '../components/second.vue'
export default{
	data(){
		return{
			message:'我们一起当前端攻城狮!'
		}
	},
	provide(){ // provide是一个匿名函数,返回一个对象
		return {
			testmethods:this.testmethods,
			message:this.message
		}
	},
	methods:{
		testmethods(){
			console.log('调用了ProvideTest这个组件')
		}
	},
	components:{
		second
	}
}
</script>
<style lang="less" scoped>
</style>

子组件:用inject接收父组件的值和方法,并且继续套一个组件

<template>
	<div id="app">
		<p>second组件:{{message}}</p>
		<third></third>
	</div>
</template>
<script>
import third from './third.vue'
export default{
	data(){
		return{
		}
	},
	inject:['message','testmethods'],
	mounted() {
		this.testmethods()
	},
	components:{
		third
	}
}
</script>
<style lang="less" scoped>
</style>

重点来了,我们称之为

孙组件:

<template>
	<div id="app">
		<p>third组件:{{message}}</p>
	</div>
</template>
<script>
export default{
	data(){
		return{
		}
	},
	//inject:['message','testmethods'], 简写
	inject:{ // 详细指定来源以及默认值
		message:{
			from:'message', //表示从组件ProvideTest传递过来的
			//default:'message' //默认值
		},
		testmethods:{
			form:'testmethods'
		}
	},
	mounted() {
		this.testmethods()
	},
}
</script>
<style lang="less" scoped>
</style>

效果下图所示

vue中provide,inject遇到的一个坑

provide、inject一般用在组件间嵌套过多,而子组件一层层的传递很麻烦,此时通过provide、inject可以跨层传递。但是最近在使用的过程中发现一个问题:

祖组件中data里的响应式数据通过provide return以后,发现孙组件无法接受到最新的值

//祖组件
<template>
    <div> this is grandparent component</div>
</template>
<script>
export {
    name:"grandparent",
    data(){
            return{
              hasMeal:false 
        }
    },
    provide(){
        return{
            hasMeal:this.hasMeal        
        }
    },
    create(){
      fetch(xxx).then(res=>{
            this.hasMeal=res.data.hasMeal  //此时是true
        })
    }
}
</script>
///
孙组件
<template>
    <div> this is grandson component</div>
</template>
<script>
export {
    name:"grandparent",
    data(){
            return{
        }
    },
    inject:['hasMeal'],
    create(){
        console.log(this.hasMeal)    //false
    }
}
</script>

hasMeal经过异步请求以后变成了true,原本期待provide最后return的值是最新的值时true,结果在孙组件页面打印this.hasMeal发现。

还是false?那是否是provide在return之前的this.hasMeal还是false呢?

经过测试发现,果不其然。进一步证明。provide里如果直接return data里的值。是不能被响应式处理的。

如何解决?

//祖组件
<template>
    <div> this is grandparent component</div>
</template>
<script>
export {
    name:"grandparent",
    data(){
            return{
              hasMeal:false 
        }
    },
    provide:()=>{
        return{
            hasMeal:this.hasMeal        
        }
    },
    create(){
      fetch(xxx).then(res=>{
            this.hasMeal=res.data.hasMeal  //此时是true
        })
    }
}
</script>
///
孙组件
<template>
    <div> this is grandson component</div>
</template>
<script>
export {
    name:"grandparent",
    data(){
            return{
        }
    },
    inject:['hasMeal'],
    create(){
        console.log(this.hasMeal())    //true
    }
}
</script>

把provide变成一个箭头函数。然后在孙页面通过执行这个函数的方式拿到函数返回的结果就可以拿到最新的值。也就间接变成了响应式的了 

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
阅读全文