在vue中,ref被用来给元素或子组件注册引用信息。引用信息将会注册在父组件的 $refs
对象上。如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子组件上,引用就指向该子组件实例。通俗的讲,ref特性就是为元素或子组件赋予一个ID引用,通过this.$refs.refName来访问元素或子组件的实例。
<template>
<div @click="startTest">
<Test ref="Tests" v-if="testShow"></Test>
</div>
</template>
<script>
import Test from './test'
export default {
data () {
return {
testShow: false
}
},
components: {
Test
},
methods: {
async startTest () {
this.testShow = true
this.$refs.Tests.submit()
console.log(this.$refs.Tests)
}
},
}
</script>
这时候打印this.$refs会undefined,如果调用子组件里面的方法,则会直接报错。
用ref 注册子组件,父组件可以通过this.$refs.xx.fn调用子组件里的函数,但是有时会出现未定义的情况,这是为什么呢?
vue 官网中ref 下有一段话 “关于 ref 注册时间的重要说明:因为 ref 本身是作为渲染结果被创建的,在初始渲染的时候你不能访问它们 - 它们还不存在!$refs 也不是响应式的,因此你不应该试图用它在模板中做数据绑定。”
也就是说 ref 只有等页面加载完成好之后你才能调用 this.$refs
,如果你使用v-if 、v-for渲染页面的话,那么在刚开始页面没没渲染之前你是拿不到this.$refs 的,所以要等到页面渲染之后拿才可以
解决办法:
1、如果你在mounted里获取this.$refs
,因为dom还未完全加载,所以你是拿不到的, update阶段则是完成了数据更新到 DOM 的阶段(对加载回来的数据进行处理),此时,就可以使用this.$refs了。
2、如果写在method中,那么可以使用 this.$nextTick(() => {}) 等页面渲染好再调用,这样就可以了。
3、或者加个定时器延时加载this.$refs
改过之后,代码如下:
<template>
<div @click="startTest">
<Test ref="Tests" v-if="testShow"></Test>
</div>
</template>
<script>
import Test from './test'
export default {
data () {
return {
testShow: false
}
},
components: {
Test
},
methods: {
async startTest () {
this.testShow = true
this.$nextTick(() => {
this.$refs.Tests.submit()
console.log(this.$refs.Tests)
})
}
},
}
</script>
成功打印了this.$refs并且调用了子组件里面的submit方法。