父组件ajax数据传递给子组件,[转]详解vue父组件传递props异步数据到子组件的问题...

本文详细探讨了在Vue中如何处理父组件异步数据传递给子组件的问题,包括在生命周期的不同阶段可能出现的问题及解决方案。通过设置v-if避免报错,使用watch监听prop变化,利用computed属性结合watch,以及使用prop默认值等方式,确保子组件能正确接收和处理父组件的异步数据。同时,文章还提到了使用事件总线(bus)进行组件间通信的方法。
摘要由CSDN通过智能技术生成

案例一

父组件parent.vue

67d0865f17d9be527254bf8a8ff6c5b4.gif

// asyncData为异步获取的数据,想传递给子组件使用

父组件

import child from './child'

export default {

data: () => ({

asyncData: ''

}),

components: {

child

},

created () {

},

mounted () {

// setTimeout模拟异步数据

setTimeout(() => {

this.asyncData = 'async data'

console.log('parent finish')

}, 2000)

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件child.vue

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件{{childData}}

export default {

props: ['childData'],

data: () => ({

}),

created () {

console.log(this.childData) // 空值

},

methods: {

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

上面按照这里的解析,子组件的html中的{{childData}}的值会随着父组件的值而改变,但是created里面的却不会发生改变(生命周期问题)

案例二

parent.vue

67d0865f17d9be527254bf8a8ff6c5b4.gif

父组件

import child from './child'

export default {

data: () => ({

asyncObject: ''

}),

components: {

child

},

created () {

},

mounted () {

// setTimeout模拟异步数据

setTimeout(() => {

this.asyncObject = {'items': [1, 2, 3]}

console.log('parent finish')

}, 2000)

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

child.vue

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件

{{childObject.items[0]}}

export default {

props: ['childObject'],

data: () => ({

}),

created () {

console.log(this.childObject) // 空值

},

methods: {

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

created里面的却不会发生改变, 子组件的html中的{{{childObject.items[0]}}的值虽然会随着父组件的值而改变,但是过程中会报错

// 首先传过来的是空,然后在异步刷新值,也开始时候childObject.items[0]等同于''.item[0]这样的操作,

所以就会报下面的错

vue.esm.js?8910:434 [Vue warn]: Error in render function:

"TypeError: Cannot read property '0' of undefined"

针对二的解决方法:

1、使用v-if可以解决报错问题,和created为空问题

67d0865f17d9be527254bf8a8ff6c5b4.gif

// parent.vue

父组件

import child from './child'

export default {

data: () => ({

asyncObject: '',

flag: false

}),

components: {

child

},

created () {

},

mounted () {

// setTimeout模拟异步数据

setTimeout(() => {

this.asyncObject = {'items': [1, 2, 3]}

this.flag = true

console.log('parent finish')

}, 2000)

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

67d0865f17d9be527254bf8a8ff6c5b4.gif

//child.vue

子组件//不报错

{{childObject.items[0]}}

export default {

props: ['childObject'],

data: () => ({

}),

created () {

console.log(this.childObject)// Object {items: [1,2,3]}

},

methods: {

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

2、子组件使用watch来监听父组件改变的prop,使用methods来代替created

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件

{{test}}

export default {

props: ['childObject'],

data: () => ({

test: ''

}),

watch: {

'childObject.items': function (n, o) {

this.test = n[0]

this.updata()

}

},

methods: {

updata () { // 既然created只会执行一次,但是又想监听改变的值做其他事情的话,只能搬到这里咯

console.log(this.test)// 1

}

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

3、子组件watch computed data 相结合,有点麻烦

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件

{{test}}

export default {

props: ['childObject'],

data: () => ({

test: ''

}),

watch: {

'childObject.items': function (n, o) {

this._test = n[0]

}

},

computed: {

_test: {

set (value) {

this.update()

this.test = value

},

get () {

return this.test

}

}

},

methods: {

update () {

console.log(this.childObject) // {items: [1,2,3]}

}

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

4、使用emit,on,bus相结合

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件

{{test}}

export default {

props: ['childObject'],

data: () => ({

test: ''

}),

created () {

// 绑定

this.$bus.on('triggerChild', (parmas) => {

this.test = parmas.items[0] // 1

this.updata()

})

},

methods: {

updata () {

console.log(this.test) // 1

}

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

这里使用了bus这个库,parent.vue和child.vue必须公用一个事件总线(也就是要引入同一个js,这个js定义了一个类似let bus = new Vue()的东西供这两个组件连接),才能相互触发

5、使用prop default来解决{{childObject.items[0]}}

67d0865f17d9be527254bf8a8ff6c5b4.gif

父组件

import child from './child'

export default {

data: () => ({

asyncObject: undefined // 这里使用null反而报0的错

}),

components: {

child

},

created () {

},

mounted () {

// setTimeout模拟异步数据

setTimeout(() => {

this.asyncObject = {'items': [1, 2, 3]}

console.log('parent finish')

}, 2000)

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

67d0865f17d9be527254bf8a8ff6c5b4.gif

子组件

{{childObject.items[0]}}

export default {

props: {

childObject: {

type: Object,

default () {

return {

items: ''

}

}

}

},

data: () => ({

}),

created () {

console.log(this.childObject) // {item: ''}

}

}

67d0865f17d9be527254bf8a8ff6c5b4.gif

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值