vue子組件通過(guò).sync修飾符修改props屬性方式
子組件通過(guò).sync修飾符修改props屬性
在vue子組件中,如果我們直接修改props中的屬性,會(huì)報(bào)錯(cuò):
Avoid mutating a prop directly since the value will be overwritten whenever the parent component re-renders. Instead, use a data or computed property based on the prop’s value. Prop being mutated: “val”
vue為我們提供了.sync修飾符來(lái)解決這個(gè)問(wèn)題:
// 父組件中,通過(guò).sync修改符綁定屬性
<my-children :val.sync='val'></my-children>
// 子組件中,修改屬性
<script>
export default {
props: ['val'],
methods: {
myClick() {
this.$emit('update:val', this.val + 1)
}
}
}
</script>
子組件修改父組件prop的幾種方式
實(shí)際工作項(xiàng)目開發(fā)中,很經(jīng)常會(huì)有父組件先傳遞值給子組件,再由子組件渲染展示的場(chǎng)景,下面我總結(jié)一下目前工作中遇到和用過(guò)的一些方式
常用方式
推薦,遵循prop單向傳遞的規(guī)則,基本數(shù)據(jù)類型和引用數(shù)據(jù)類型均可。
1. 通過(guò)父組件on監(jiān)聽子組件emit事件實(shí)現(xiàn)修改prop
原理:
- 給子組件中的input標(biāo)簽綁定value屬性賦值prop中的值,因?yàn)槭怯胿alue而不是v-model,所以修改input值的時(shí)候并不會(huì)影響到prop。
- 然后給input綁定input事件,每次輸入內(nèi)容都會(huì)觸發(fā)input事件,在事件里通過(guò)this.$emit(‘父要監(jiān)聽的事件名’, 修改后的值)去將值向上傳遞。
- 父組件中通過(guò)on監(jiān)聽子組件剛剛觸發(fā)的emit事件,將事件傳遞過(guò)來(lái)的值更新到父組件的data中。
- 父組件data中的值更新后,因?yàn)橛型ㄟ^(guò)prop傳遞給子組件,所以子組件也會(huì)同步更新prop值并渲染視圖層。

父組件代碼如下:
<template>
? <div style="background-color: skyblue;">
? ? <h3>通過(guò)父組件on監(jiān)聽子組件emit事件實(shí)現(xiàn)修改prop</h3>
? ? <div>父obj:{{ obj }}</div>
? ? <div>父msg:{{ msg }}</div>
? ? <!-- 父組件調(diào)用子組件時(shí)通過(guò)on監(jiān)聽子組件觸發(fā)的emit事件,以接收值并更新用 -->
? ? <emitChild
? ? ? :obj="obj"
? ? ? :msg="msg"
? ? ? @update-obj="updateObj"
? ? ? @update-msg="updateMsg"
? ? />
? </div>
</template><script>
import emitChild from './components/emitChild'
export default {
? name: 'emitUpdate',
? components: {
? ? emitChild
? },
? data () {
? ? return {
? ? ? obj: {
? ? ? ? name: 'zhangsan',
? ? ? ? age: 18
? ? ? },
? ? ? msg: 'hello'
? ? }
? },
? methods: {
? ? // 監(jiān)聽子組件觸發(fā)的事件并更新data中的obj
? ? updateObj (key, newVal) {
? ? ? this.obj[key] = newVal
? ? },
? ? // 監(jiān)聽子組件觸發(fā)的事件并更新data中的msg
? ? updateMsg (newVal) {
? ? ? this.msg = newVal
? ? }
? }
}
</script>子組件代碼如下:
<template> ? <div style="background-color: pink;"> ? ? <div> ? ? ? <span>修改name:</span> ? ? ? <!-- 這里綁定值用value,因?yàn)閕nput在這主要用作展示prop數(shù)據(jù),實(shí)際修改不在子組件,子組件只是作為修改觸發(fā)源 --> ? ? ? <!-- 綁定input事件,作為觸發(fā)源的入口 --> ? ? ? <input type="text" :value="obj.name" @input="updateObj($event, 'name')"> ? ? </div> ? ? <div> ? ? ? <span>修改age:</span> ? ? ? <input type="text" :value="obj.age" @input="updateObj($event, 'age')"> ? ? </div> ? ? <div> ? ? ? <span>修改msg:</span> ? ? ? <input type="text" :value="msg" @input="updateMsg($event.target.value)"> ? ? </div> ? </div> </template>
<script>
export default {
? name: 'emitUpdateChild',
? props: {
? ? obj: {
? ? ? type: Object,
? ? ? default: () => {}
? ? },
? ? msg: {
? ? ? type: String,
? ? ? default: ''
? ? }
? },
? methods: {
? ? // 通知父組件更新obj
? ? updateObj ($event, key) {
? ? ? // 接收輸入的值,和obj中對(duì)應(yīng)需要更新值的屬性,然后回傳給父組件
? ? ? // 父組件就可以知道子組件需要更新obj中哪個(gè)屬性的值
? ? ? this.$emit('update-obj', key, $event.target.value)
? ? },
? ? // 通知父組件更新msg
? ? updateMsg (newVal) {
? ? ? this.$emit('update-msg', newVal)
? ? }
? }
}
</script>2. 通過(guò)父組件sync修飾符 + 子組件emit事件實(shí)現(xiàn)修改prop
原理:
- 給子組件中的input標(biāo)簽綁定value屬性賦值prop中的值,因?yàn)槭怯胿alue而不是v-model,所以修改input值的時(shí)候并不會(huì)影響到prop。
- 然后給input綁定input事件,每次輸入內(nèi)容都會(huì)觸發(fā)input事件,在事件里通過(guò)this.$emit(‘父要監(jiān)聽的事件名’, 修改后的值)去將值向上傳遞。
- 父組件調(diào)用子組件傳遞prop時(shí),在prop后加上.sync即可將事件傳遞過(guò)來(lái)的值更新到父組件的data中,因?yàn)閟ync其實(shí)就相當(dāng)于執(zhí)行了@監(jiān)聽子觸發(fā)的事件名 = "父data中的屬性名 = emit傳遞的值(即$event)"這一段代碼。
- 父組件data中的值更新后,因?yàn)橛型ㄟ^(guò)prop傳遞給子組件,所以子組件也會(huì)同步更新prop值并渲染視圖層。

父組件代碼如下:
<template>
? <div style="background-color: skyblue;">
? ? <h3>通過(guò)父組件sync修飾符 + 子組件emit事件實(shí)現(xiàn)修改prop</h3>
? ? <div>父obj:{{ obj }}</div>
? ? <div>父msg:{{ msg }}</div>
? ? <!-- 父組件調(diào)用子組件傳遞prop時(shí),在prop后加上.sync即可 -->
? ? <syncChild :obj.sync="obj" :msg.sync="msg" />
? ? <!--
? ? ? sync其實(shí)就相當(dāng)于執(zhí)行了
? ? ? ? @監(jiān)聽子觸發(fā)的事件名 = "父data中的屬性名 = emit傳遞的值(即$event)"
? ? ? 這一段代碼
? ? -->
? ? <!-- 效果相當(dāng)于下面的代碼,所以父組件methods中不需要再定義修改data數(shù)據(jù)的方法 -->
? ? <!--
? ? ? <syncChild
? ? ? ? :obj="obj"
? ? ? ? :msg="msg"
? ? ? ? @update-obj="obj = $event"
? ? ? ? @update-msg="msg = $event"
? ? ? />
? ? -->
? </div>
</template><script>
import syncChild from './components/syncChild'
export default {
? name: 'syncUpdate',
? components: {
? ? syncChild
? },
? data () {
? ? return {
? ? ? obj: {
? ? ? ? name: 'zhangsan',
? ? ? ? age: 18
? ? ? },
? ? ? msg: 'hello'
? ? }
? }
}
</script>子組件代碼如下:
<template> ? <div style="background-color: pink;"> ? ? <div> ? ? ? <span>修改name:</span> ? ? ? <!-- 這里綁定值用value,因?yàn)閕nput在這主要用作展示prop數(shù)據(jù),實(shí)際修改不在子組件,子組件只是作為修改觸發(fā)源 --> ? ? ? <!-- 綁定input事件,作為觸發(fā)源的入口 --> ? ? ? <input type="text" :value="childObj.name" @input="updateObj($event, 'name')"> ? ? </div> ? ? <div> ? ? ? <span>修改age:</span> ? ? ? <input type="text" :value="childObj.age" @input="updateObj($event, 'age')"> ? ? </div> ? ? <div> ? ? ? <span>修改msg:</span> ? ? ? <input type="text" :value="msg" @input="updateMsg($event.target.value)"> ? ? </div> ? </div> </template>
<script>
// 這里引入lodash工具庫(kù)的cloneDeep深拷貝方法
// 官方文檔地址 https://www.lodashjs.com/
import { cloneDeep } from 'lodash'
export default {
? name: 'emitUpdateChild',
? props: {
? ? obj: {
? ? ? type: Object,
? ? ? default: () => {}
? ? },
? ? msg: {
? ? ? type: String,
? ? ? default: ''
? ? }
? },
? data () {
? ? return {
? ? ? // 這里通過(guò)深拷貝將prop的obj復(fù)制了一份,主要為了:
? ? ? // 1.區(qū)分2個(gè)對(duì)象(引用類型)用的不是同一個(gè)內(nèi)存地址
? ? ? // 2.保留對(duì)象中所有的值,方便子組件渲染/修改/回傳用
? ? ? childObj: cloneDeep(this.obj)
? ? }
? },
? methods: {
? ? // 通知父組件更新obj
? ? updateObj ($event, key) {
? ? ? // 接收輸入的值,和childObj中對(duì)應(yīng)需要更新值的屬性
? ? ? // 然后更新childOBj后,回傳給父組件
? ? ? // 父組件直接把拿到的childObj更新到data的obj即可
? ? ? this.childObj[key] = $event.target.value
? ? ? this.$emit('update:obj', this.childObj)
? ? },
? ? // 通知父組件更新msg
? ? updateMsg (newVal) {
? ? ? this.$emit('update:msg', newVal)
? ? }
? }
}
</script>取巧方式
主要針對(duì)引用數(shù)據(jù)類型,繞過(guò)了vue對(duì)于prop的檢測(cè)機(jī)制,具體看項(xiàng)目規(guī)范是否允許這樣寫。
3.通過(guò)data實(shí)現(xiàn)修改prop
前提:只有引用數(shù)據(jù)類型可以實(shí)現(xiàn)
原理:
- 將父組件傳入的prop直接賦值給子組件的data,此時(shí)prop和data兩邊的變量指向的都是同一個(gè)內(nèi)存地址,所以修改data等于修改prop。
- vue2開始不允許直接修改prop,但此時(shí)我們表面修改的是data不是prop,因此vue不會(huì)拋出錯(cuò)誤,相當(dāng)于繞過(guò)了vue對(duì)于不允許修改prop的檢測(cè)機(jī)制。

父組件代碼如下:
<template>
? <div style="background-color: skyblue;">
? ? <h3>通過(guò)賦值到data實(shí)現(xiàn)修改prop</h3>
? ? <div>父obj:{{ obj }}</div>
? ? <div>父msg:{{ msg }}</div>
? ? <dataChild :obj="obj" :msg.sync="msg" />
? </div>
</template><script>
import dataChild from './components/dataChild'
export default {
? name: 'dataUpdate',
? components: {
? ? dataChild
? },
? data () {
? ? return {
? ? ? obj: {
? ? ? ? name: 'zhangsan',
? ? ? ? age: 18
? ? ? },
? ? ? msg: 'hello'
? ? }
? }
}
</script>子組件代碼如下:
<template>
<div style="background-color: pink;">
<div>
<span>修改name:</span>
<!-- 這里因?yàn)槲覀冎苯影裵rop賦值給data了,所以可以直接用v-model雙向綁定修改數(shù)據(jù) -->
<input type="text" v-model="dataObj.name">
</div>
<div>
<span>修改age:</span>
<input type="text" v-model="dataObj.age">
</div>
<div>
<span>修改msg:</span>
<!-- 這里提供另一種修改基本數(shù)據(jù)類型的思路,可以通過(guò)watch偵聽器實(shí)現(xiàn) -->
<input type="text" v-model="dataMsg">
</div>
</div>
</template>
<script>
export default {
name: 'dataChild',
props: {
obj: {
type: Object,
default: () => {}
},
msg: {
type: String,
default: ''
}
},
data () {
return {
// 引用數(shù)據(jù)類型直接賦值時(shí)是淺拷貝,只會(huì)復(fù)制內(nèi)存地址,修改其中一個(gè)會(huì)影響另一個(gè)
dataObj: this.obj,
// 基本數(shù)據(jù)類型直接賦值時(shí)是復(fù)制值,修改其中一個(gè)不會(huì)影響另一個(gè)
dataMsg: this.msg
}
},
watch: {
// 這里偵聽data中復(fù)制過(guò)來(lái)的dataMsg,當(dāng)修改時(shí)執(zhí)行emit通知父組件進(jìn)行更新
dataMsg (newVal) {
this.$emit('update:msg', newVal)
}
}
}
</script>
4.通過(guò)計(jì)算屬性computed實(shí)現(xiàn)修改prop
前提:只有引用數(shù)據(jù)類型可以實(shí)現(xiàn)
原理:
- 在子組件中直接通過(guò)計(jì)算屬性computed監(jiān)聽父組件傳入的prop,此時(shí)計(jì)算屬性computed和prop兩邊的變量指向的都是同一個(gè)內(nèi)存地址,所以修改計(jì)算屬性computed等于修改prop。
- vue2開始不允許直接修改prop,但此時(shí)我們表面修改的是計(jì)算屬性computed不是prop,因此vue不會(huì)拋出錯(cuò)誤,相當(dāng)于繞過(guò)了vue對(duì)于不允許修改prop的檢測(cè)機(jī)制。

父組件代碼如下:
<template>
? <div style="background-color: skyblue;">
? ? <h3>通過(guò)計(jì)算屬性監(jiān)聽實(shí)現(xiàn)修改prop</h3>
? ? <div>父obj:{{ obj }}</div>
? ? <div>父msg:{{ msg }}</div>
? ? <computedChild :obj="obj" :msg.sync="msg" />
? </div>
</template><script>
import computedChild from './components/computedChild'
export default {
? name: 'computedUpdate',
? components: {
? ? computedChild
? },
? data () {
? ? return {
? ? ? obj: {
? ? ? ? name: 'zhangsan',
? ? ? ? age: 18
? ? ? },
? ? ? msg: 'hello'
? ? }
? }
}
</script>子組件代碼如下:
<template> ? <div style="background-color: pink;"> ? ? <div> ? ? ? <span>修改name:</span> ? ? ? <!-- 這里因?yàn)槲覀冎苯油ㄟ^(guò)計(jì)算屬性computed監(jiān)聽prop了,所以可以直接用v-model雙向綁定修改數(shù)據(jù) --> ? ? ? <input type="text" v-model="computedObj.name"> ? ? </div> ? ? <div> ? ? ? <span>修改age:</span> ? ? ? <input type="text" v-model="computedObj.age"> ? ? </div> ? ? <div> ? ? ? <span>修改msg:</span> ? ? ? <!-- 這里提供另一種修改基本數(shù)據(jù)類型的思路,可以通過(guò)計(jì)算屬性computed的setter實(shí)現(xiàn) --> ? ? ? <input type="text" v-model="computedMsg"> ? ? </div> ? </div> </template>
<script>
export default {
? name: 'computedChild',
? props: {
? ? obj: {
? ? ? type: Object,
? ? ? default: () => {}
? ? },
? ? msg: {
? ? ? type: String,
? ? ? default: ''
? ? }
? },
? computed: {
? ? computedObj () {
? ? ? // 這里直接return引用數(shù)據(jù)類型obj,此時(shí)computedObj相當(dāng)于obj
? ? ? // 所以是淺拷貝,只會(huì)復(fù)制內(nèi)存地址,修改其中一個(gè)會(huì)影響另一個(gè)
? ? ? return this.obj
? ? },
? ? computedMsg: {
? ? ? get () {
? ? ? ? // 這里prop每次更新時(shí),都會(huì)觸發(fā)計(jì)算屬性的getter方法獲取最新的值
? ? ? ? // 直接return基本數(shù)據(jù)類型時(shí)是復(fù)制值,修改其中一個(gè)不會(huì)影響另一個(gè)
? ? ? ? return this.msg
? ? ? },
? ? ? set (newVal) {
? ? ? ? // 這里利用計(jì)算屬性的setter方法,監(jiān)聽到值修改時(shí)觸發(fā)emit通知父組件更新值
? ? ? ? this.$emit('update:msg', newVal)
? ? ? }
? ? }
? }
}
</script>以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Vue 自定義標(biāo)簽的src屬性不能使用相對(duì)路徑的解決
這篇文章主要介紹了Vue 自定義標(biāo)簽的src屬性不能使用相對(duì)路徑的解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
前端使用vue實(shí)現(xiàn)token無(wú)感刷新的三種方案解析
這篇文章主要為大家介紹了前端使用vue實(shí)現(xiàn)token無(wú)感刷新的三種方案詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
如何測(cè)量vue應(yīng)用運(yùn)行時(shí)的性能
這篇文章主要介紹了如何測(cè)量vue應(yīng)用運(yùn)行時(shí)的性能,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,,需要的朋友可以參考下2019-06-06
vant的Loading加載動(dòng)畫組件的使用(通過(guò)接口拿數(shù)據(jù)時(shí)顯示加載狀態(tài))
這篇文章主要介紹了vant的Loading加載動(dòng)畫組件的使用,通過(guò)接口拿數(shù)據(jù)時(shí)顯示加載狀態(tài),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01
Vue查詢數(shù)據(jù)并通過(guò)bootstarp?table渲染數(shù)據(jù)
這篇文章主要為大家介紹了Vue查詢數(shù)據(jù)并通過(guò)bootstarp?table渲染數(shù)據(jù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04

