一文詳解Vue3中的14種組件通信方式
對(duì)于日常使用vue3開(kāi)發(fā)項(xiàng)目的前端小伙伴來(lái)說(shuō),組件通信方式可以說(shuō)是必會(huì)的基本功,今天帶大家一起盤(pán)下vue3的通信方式。
我們按照組件的關(guān)系來(lái)劃分。總共包含14中組件通信方式。
一:父子通信
1.1、父?jìng)髯樱簆rops
最最常用的通信方式是props了,父組件通過(guò)props方式將屬性傳遞給子組件,子組件接受props并用于數(shù)據(jù)操作和頁(yè)面渲染。
注意:子組件不要直接修改父組件傳遞過(guò)來(lái)的props,保持自上而下單項(xiàng)數(shù)據(jù)流,這樣會(huì)讓數(shù)據(jù)的流向十分清晰,方便后續(xù)維護(hù)!
// Parent.vue
<template>
<Child :msg="msg"/>
</template>
<script setup>
import { ref } from 'vue';
import Child from './components/Child.vue';
const msg = ref('hello world');
</script>
// Child.vue
<template>
<div>
propsData:{{ msg }}
</div>
</template>
<script setup>
defineProps({
msg: String
})
</script>
1.2、子傳父:defineEmits
通過(guò)defineEmits可以讓子組件的值傳遞到父組件中。
其用法如下:
先在子組件中用defineEmits([...emitName])定義一個(gè)或多個(gè)emit,它的返回值是一個(gè)emits函數(shù),然后可以通過(guò)調(diào)用emits函數(shù)向父組件發(fā)射時(shí)間,并攜帶參數(shù)。
// Child.vue
<template>
<div @click="onClick">
child
</div>
</template>
<script setup>
const emits = defineEmits(['update']);
const onClick = () => {
emits('update', 'child update');
}
</script>
在父組件中通過(guò)@符 + 事件名監(jiān)聽(tīng)子組件發(fā)射 出來(lái)的事件,并接收其傳過(guò)來(lái)的值。
// Parent.vue
<template>
<div>
Parent
<Child @update="update"/>
</div>
</template>
<script setup>
import Child from './Child.vue';
const update = val => {
console.log(val); // 當(dāng)子組件點(diǎn)擊事件觸發(fā)后,這里會(huì)打印 child update
}
</script>
在vue2的組件中還可以通過(guò)this.$on、this.$emit來(lái)監(jiān)聽(tīng)、發(fā)射事件,以達(dá)到傳值的目的,但在vue3中已廢棄這種寫(xiě)法。
1.3、$attrs
如果需要在子組件中接收的props很多,如果在props聲明比較繁瑣,所以vue給我們提供了一個(gè)優(yōu)雅的解決方案,即$attrs,$attrs指的是父組件傳遞給子組件的所有屬性中,剔除在props中定義的那部分之后,剩下的就會(huì)放在$attrs中。
舉個(gè)例子:
// Parent.vue
<template>
<div>
Parent
<Child :msg1="1" :msg2="2" />
</div>
</template>
這里父組件給子組件傳遞了兩個(gè)屬性msg1、msg2。
// Child.vue
<template>
<div>
child: {{ $attrs }}
</div>
</template>
<script setup>
defineProps({
msg1: String
})
</script>
這里子組件使用了defineProps定義了msg1,則頁(yè)面中$attrs的值為{ msg2: 2 }。
還可以使用v-bind將$attrs的所有數(shù)據(jù),以屬性的方式全部傳遞到子組件中,我們平常在封裝組件的時(shí)候,這個(gè)東西就能幫助我們實(shí)現(xiàn)組件的屬性透?jìng)?,十分好用?/p>
<template>
<Child v-bind="$attrs"/>
</template>
注意:在vue3中$listeners已廢棄,無(wú)法使用。
1.4、$ref + defineExpose
通過(guò)$ref可以拿到組件的實(shí)例,defineExpose可以顯式指定在 <script setup> 組件中要暴露出去的屬性,它兩一起配合使用,就能實(shí)現(xiàn)父子組件的通信。
其用法如下:
在子組件中通過(guò)defineExpose暴露一個(gè)update方法。
// Child.vue
<script setup>
defineExpose({
update(val) {
console.log('父組件傳遞過(guò)來(lái)的值', val);
}
})
</script>
在父組件中通過(guò)ref拿到組件實(shí)例并調(diào)用子組件暴露的update方法。
// Parent.vue
<template>
<div>
Parent
<Child ref="childRef"/>
</div>
</template>
<script setup>
import Child from './Child.vue';
import { ref, onMounted } from 'vue';
const childRef = ref(null);
onMounted(() => {
childRef.value.update('hello')
})
const update = () => {}
</script>
1.5、$parent
$parent代表當(dāng)前組件的父組件實(shí)例,如果當(dāng)前組件是頂層組件,則$parent的值為null。
我們可以通過(guò)$parent拿到父組件的實(shí)例,自然就可以進(jìn)行父子組件的交互了。一般也是和defineExpose配合使用,和$ref + defineExpose用法類(lèi)似,這里就不多說(shuō)了。
注意:$children在vue3中已經(jīng)廢棄,無(wú)法使用。
1.6、作用域插槽
通過(guò)作用域插槽可以實(shí)現(xiàn)子組件向父組件傳遞數(shù)據(jù)。
子組件代碼:
<template>
<div>
<slot :data="{ a:1, b: 2 }"/>
</div>
</template>
子組件可以在slot標(biāo)簽上傳遞數(shù)據(jù)給父組件。
父組件代碼:
<template>
<Child>
<template v-slot="slotProps">
{{ slotProps.data }}
</template>
</Child>
</template>
<script setup>
import Child from './Child.vue';
</script>
父組件用v-slot來(lái)接收數(shù)據(jù),并渲染到頁(yè)面上。
1.7、v-model
v-model可以在組件上使用以實(shí)現(xiàn)雙向綁定,vue內(nèi)部會(huì)幫你傳遞值和綁定事件,也是達(dá)到了父子組件通訊的效果。
從vue3.4開(kāi)始,還可以使用defineModel便利宏,其用法如下:
子組件代碼:
// Child.vue
<template>
<div>model的值: {{ model }}</div>
<button @click="handleClick">+1</button>
</template>
<script setup>
const model = defineModel()
function handleClick() {
model.value++
}
</script>
父組件代碼:
// Parent.vue
<template>
<Child1 v-model="modelValue"></Child1>
Parent:{{ modelValue }}
</template>
<script setup>
import Child from './Child.vue';
const modelValue = ref(0)
</script>
defineModel的返回值就是一個(gè)ref,你可以隨意訪問(wèn)和修改它,并且它會(huì)和父組件的v-model綁定的值保持同步,也就是實(shí)現(xiàn)了雙向綁定。
二、兄弟組件
兩個(gè)兄弟關(guān)系組件進(jìn)行通信,我們一般會(huì)借助第三方媒介。
2.1、mitt
mitt相當(dāng)于我們vue2的事件總線$bus,只是vue3將其廢棄,所以我們借助mitt實(shí)現(xiàn)類(lèi)似$bus的效果。
用法如下:
安裝mitt
npm install mitt
初始化mitt
// emitter.js import mitt from'mitt'; export default mitt();
兄弟組件1:
<script setup>
import emitter from '@/utils/emitter'
emitter.on('update', (val) => {
console.log('update事件觸發(fā)', val)
})
</script>
兄弟組件2:
<script setup>
import emitter from '@/utils/emitter'
setTimeout(() => {
emitter.emit('update', 'hello')
}, 1000)
</script>
2.2、$parent
我們可以把狀態(tài)(即數(shù)據(jù))定義在父組件中,兩個(gè)兄弟組件可以借助其共同的父組件共享同一份數(shù)據(jù),間接實(shí)現(xiàn)通信。
2.3、vuex/pinia
vuex是vue官方提供的狀態(tài)管理工具,用它可以實(shí)現(xiàn)全局的狀態(tài)共享,自然也可以實(shí)現(xiàn)兄弟組件的通信了。當(dāng)然也可以使用pinia替代vuex。
2.4、app.config.globalProperties
app.config.globalProperties是一個(gè)全局的對(duì)象,在應(yīng)用內(nèi)所有組件實(shí)例都能訪問(wèn)到,當(dāng)組件屬性名和它發(fā)生同名沖突時(shí),采取就近原則,以組件的為準(zhǔn),這個(gè)就相當(dāng)于vue2的Vue.prototype。
三、跨層級(jí)通信
3.1、mitt
mitt可以實(shí)現(xiàn)全局的通信,這個(gè)在上面介紹兄弟組件通信的時(shí)候已經(jīng)說(shuō)過(guò),這里不再多說(shuō)了。
3.2、vuex/pinia
vuex和pinia都是全局的狀態(tài)管理工具,跨層級(jí)通信也不再話下。
3.3、provide/inject
provide/inject是vue3提供的可以跨層級(jí)通信的方式。
其用法如下:
在父組件/根組件中定義provide,提供數(shù)據(jù)。
// App.vue
<script setup>
import { ref, provide } from 'vue';
const name = ref('sam');
provide('name', name)
</script>
在子組件/孫子組件中使用inject,注入數(shù)據(jù)。
// 后代組件
<script setup>
import { inject } from 'vue';
const name = inject('name');
console.log('name', name.value); // 輸出name
</script>
四、其它方式
4.1、瀏覽器本地存儲(chǔ)storage
html5提供了一套storage API,包括了localStorage和sessionStorage,它實(shí)現(xiàn)持久化存儲(chǔ)、緩存等功能,自然也可以用來(lái)組件間通信了。
// 組件A
<script setup>
sessionStorage.setItem('name', 'jack');
</script>
// 組件B
<script setup>
setTimeout(() => {
console.log(sessionStorage.getItem('name')); //打印 jack
}, 1000)
</script>
這里我在組件A使用sessionStarge設(shè)置了一個(gè)name值,在組件B里面就能拿到了,只要保證獲取值在設(shè)置值之后執(zhí)行就行了。
4.2、全局window對(duì)象(不推薦使用)
window作為一個(gè)全局對(duì)象,當(dāng)然也可以使用它來(lái)通信了,不過(guò)它既然誰(shuí)都可以訪問(wèn)到,就存在如下問(wèn)題:
- 命令沖突問(wèn)題;
- 難以追蹤數(shù)據(jù)修改,可維護(hù)性差;
- 掛在window上的數(shù)據(jù)難以銷(xiāo)毀,從而造成內(nèi)存泄漏。
所以不推薦使用window對(duì)象進(jìn)行通信。
4.3、 ES6模塊化import/export
我們可以使用ES5的模塊化規(guī)范import/export實(shí)現(xiàn)通信。
// a.js
export let a = undefined;
setTimeout(() => {
a = 1;
}, 1000)
// b.js
import { a } from './a.js'
setTimeout(() => {
console.log(a); // 打印1
}, 2000)
由于ES module采用的是符號(hào)綁定,所以就算export的值是一個(gè)基本數(shù)據(jù)類(lèi)型的值,后續(xù)修改了也能訪問(wèn)到。
以上就是一文詳解Vue3中的14種組件通信方式的詳細(xì)內(nèi)容,更多關(guān)于Vue3組件通信方式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
在Vue3項(xiàng)目中安裝和配置Three.js的操作代碼
Three.js是一個(gè)輕量級(jí)的WebGL封裝庫(kù),用于在瀏覽器中渲染復(fù)雜的3D圖形,它提供了便捷的API,可以快速構(gòu)建3D場(chǎng)景、對(duì)象和動(dòng)畫(huà),Vue.js是一個(gè)漸進(jìn)式JavaScript框架,擅長(zhǎng)構(gòu)建用戶(hù)界面,本文給大家介紹了在Vue3項(xiàng)目中安裝和配置Three.js的操作,需要的朋友可以參考下2024-12-12
Vue實(shí)用功能之實(shí)現(xiàn)拖拽元素、列表拖拽排序
在日常開(kāi)發(fā)中,特別是管理端,經(jīng)常會(huì)遇到要實(shí)現(xiàn)拖拽排序的效果,下面這篇文章主要給大家介紹了關(guān)于Vue實(shí)用功能之實(shí)現(xiàn)拖拽元素、列表拖拽排序的相關(guān)資料,需要的朋友可以參考下2022-10-10
Vuejs開(kāi)發(fā)環(huán)境搭建及熱更新【推薦】
Vue.js是目前很火的一個(gè)前端框架,采用MVVM模式設(shè)計(jì),它是以數(shù)據(jù)驅(qū)動(dòng)和組件化的思想構(gòu)建的。本文重點(diǎn)給大家介紹Vuejs開(kāi)發(fā)環(huán)境搭建及熱更新的相關(guān)知識(shí),需要的朋友參考下吧2018-09-09
Vue3快速實(shí)現(xiàn)文件上傳OSS的方法詳解
這篇文章給大家介紹了Vue3快速實(shí)現(xiàn)文件上傳OSS的方法,上傳文件可以說(shuō)是經(jīng)典的需求了,在后臺(tái)管理項(xiàng)目中隨處可見(jiàn),一般是由前端進(jìn)行文件上傳,然后再由后端去處理,本文旨在實(shí)現(xiàn)上傳功能,不考慮額外的功能(如文件尺寸限制),感興趣的朋友可以參考下2024-01-01
vue中調(diào)用HTTP請(qǐng)求的詳細(xì)步驟
這篇文章主要介紹了vue中調(diào)用HTTP請(qǐng)求的詳細(xì)步驟,文中通過(guò)代碼示例給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定幫助,需要的朋友可以參考下2024-07-07
vue實(shí)現(xiàn)微信公眾號(hào)h5跳轉(zhuǎn)小程序的示例代碼
本文主要介紹了vue實(shí)現(xiàn)微信公眾號(hào)h5跳轉(zhuǎn)小程序的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
VSCode使React?Vue代碼調(diào)試變得更爽
這篇文章主要為大家介紹了VSCode使React?Vue代碼調(diào)試變得更爽的使用方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
vue滾動(dòng)固定頂部及修改樣式的實(shí)例代碼
這篇文章主要介紹了vue滾動(dòng)固定頂部及修改樣式,本文給大家提到了滾動(dòng)固定位置有多種方法,感興趣的朋友跟隨小編一起看看吧2019-05-05

