vue的 this.$refs 打印为undefined解决办法

在vue中,ref被用来给元素或子组件注册引用信息。引用信息将会注册在父组件的 r e f s 对象上。如果在普通的 D O M 元素上使用,引用指向的就是 D O M 元素;如果用在子组件上,引用就指向该子组件实例。通俗的讲, r e f 特性就是为元素或子组件赋予一个 I D 引用 , 通过 t h i s . refs对象上。如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子组件上,引用就指向该子组件实例。通俗的讲,ref特性就是为元素或子组件赋予一个ID引用,通过this.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>

image.png
这时候打印this.$refs会undefined,如果调用子组件里面的方法,则会直接报错。

用ref 注册子组件,父组件可以通过this.$refs.xx.fn调用子组件里的函数,但是有时会出现未定义的情况,这是为什么呢?

vue 官网中ref 下有一段话  “关于 ref 注册时间的重要说明:因为 ref 本身是作为渲染结果被创建的,在初始渲染的时候你不能访问它们 - 它们还不存在!$refs 也不是响应式的,因此你不应该试图用它在模板中做数据绑定。”

也就是说 ref 只有等页面加载完成好之后你才能调用 this.r e f s ,如果你使用 v − i f 、 v − f o r 渲染页面的话,那么在刚开始页面没没渲染之前你是拿不到 t h i s . refs ,如果你使用v-if 、v-for渲染页面的话,那么在刚开始页面没没渲染之前你是拿不到this.refs,如果你使用vifvfor渲染页面的话,那么在刚开始页面没没渲染之前你是拿不到this.refs 的,所以要等到页面渲染之后拿才可以

解决办法:

1、如果你在mounted里获取this.r e f s ,因为 d o m 还未完全加载,所以你是拿不到的,  u p d a t e 阶段则是完成了数据更新到 D O M 的阶段 ( 对加载回来的数据进行处理 ) ,此时,就可以使用 t h i s . refs,因为dom还未完全加载,所以你是拿不到的, update阶段则是完成了数据更新到 DOM 的阶段(对加载回来的数据进行处理),此时,就可以使用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>

image.png
成功打印了this.$refs并且调用了子组件里面的submit方法。


版权声明:本文为weixin_45658814原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。