亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

一文詳解Vue3中的14種組件通信方式

 更新時(shí)間:2025年01月10日 10:46:41   作者:程序員小寒  
對(duì)于日常使用vue3開發(fā)項(xiàng)目的前端小伙伴來(lái)說(shuō),組件通信方式可以說(shuō)是必會(huì)的基本功,今天帶大家一起盤下vue3中的14種通信方式,希望對(duì)大家有所幫助

對(duì)于日常使用vue3開發(fā)項(xiàng)目的前端小伙伴來(lái)說(shuō),組件通信方式可以說(shuō)是必會(huì)的基本功,今天帶大家一起盤下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)聽子組件發(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.$onthis.$emit來(lái)監(jiān)聽、發(fā)射事件,以達(dá)到傳值的目的,但在vue3中已廢棄這種寫法。

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用法類似,這里就不多說(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開始,還可以使用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)類似$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

vuexvue官方提供的狀態(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)于vue2Vue.prototype。

三、跨層級(jí)通信

3.1、mitt

mitt可以實(shí)現(xiàn)全局的通信,這個(gè)在上面介紹兄弟組件通信的時(shí)候已經(jīng)說(shuō)過(guò),這里不再多說(shuō)了。

3.2、vuex/pinia

vuexpinia都是全局的狀態(tài)管理工具,跨層級(jí)通信也不再話下。

3.3、provide/inject

provide/injectvue3提供的可以跨層級(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,包括了localStoragesessionStorage,它實(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ù)難以銷毀,從而造成內(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ù)類型的值,后續(xù)修改了也能訪問(wèn)到。

以上就是一文詳解Vue3中的14種組件通信方式的詳細(xì)內(nèi)容,更多關(guān)于Vue3組件通信方式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 在Vue3項(xiàng)目中安裝和配置Three.js的操作代碼

    在Vue3項(xiàng)目中安裝和配置Three.js的操作代碼

    Three.js是一個(gè)輕量級(jí)的WebGL封裝庫(kù),用于在瀏覽器中渲染復(fù)雜的3D圖形,它提供了便捷的API,可以快速構(gòu)建3D場(chǎng)景、對(duì)象和動(dòng)畫,Vue.js是一個(gè)漸進(jìn)式JavaScript框架,擅長(zhǎng)構(gòu)建用戶界面,本文給大家介紹了在Vue3項(xiàng)目中安裝和配置Three.js的操作,需要的朋友可以參考下
    2024-12-12
  • JavaScript 沙箱探索

    JavaScript 沙箱探索

    這篇文章主要介紹了JavaScript 沙箱探索,沙箱是基于 event bus 形式的通信實(shí)現(xiàn)上層的功能,文章的例子選擇接口實(shí)現(xiàn)了 web worker 與 quickjs 的 EventEmitter,,需要的朋友可以參考一下
    2021-10-10
  • Vue實(shí)用功能之實(shí)現(xiàn)拖拽元素、列表拖拽排序

    Vue實(shí)用功能之實(shí)現(xiàn)拖拽元素、列表拖拽排序

    在日常開發(fā)中,特別是管理端,經(jīng)常會(huì)遇到要實(shí)現(xiàn)拖拽排序的效果,下面這篇文章主要給大家介紹了關(guān)于Vue實(shí)用功能之實(shí)現(xiàn)拖拽元素、列表拖拽排序的相關(guān)資料,需要的朋友可以參考下
    2022-10-10
  • Vuejs開發(fā)環(huán)境搭建及熱更新【推薦】

    Vuejs開發(fā)環(huán)境搭建及熱更新【推薦】

    Vue.js是目前很火的一個(gè)前端框架,采用MVVM模式設(shè)計(jì),它是以數(shù)據(jù)驅(qū)動(dòng)和組件化的思想構(gòu)建的。本文重點(diǎn)給大家介紹Vuejs開發(fā)環(huán)境搭建及熱更新的相關(guān)知識(shí),需要的朋友參考下吧
    2018-09-09
  • vue中的mescroll搜索運(yùn)用及各種填坑處理

    vue中的mescroll搜索運(yùn)用及各種填坑處理

    這篇文章主要介紹了vue中的mescroll搜索運(yùn)用及各種填坑處理,文中通過(guò)代碼給大家講解了mescroll vue使用,感興趣的朋友跟隨小編一起看看吧
    2019-10-10
  • Vue3快速實(shí)現(xiàn)文件上傳OSS的方法詳解

    Vue3快速實(shí)現(xiàn)文件上傳OSS的方法詳解

    這篇文章給大家介紹了Vue3快速實(shí)現(xiàn)文件上傳OSS的方法,上傳文件可以說(shuō)是經(jīng)典的需求了,在后臺(tái)管理項(xiàng)目中隨處可見,一般是由前端進(jìn)行文件上傳,然后再由后端去處理,本文旨在實(shí)現(xiàn)上傳功能,不考慮額外的功能(如文件尺寸限制),感興趣的朋友可以參考下
    2024-01-01
  • vue中調(diào)用HTTP請(qǐng)求的詳細(xì)步驟

    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)小程序的示例代碼

    本文主要介紹了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)試變得更爽

    這篇文章主要為大家介紹了VSCode使React?Vue代碼調(diào)試變得更爽的使用方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • vue滾動(dòng)固定頂部及修改樣式的實(shí)例代碼

    vue滾動(dòng)固定頂部及修改樣式的實(shí)例代碼

    這篇文章主要介紹了vue滾動(dòng)固定頂部及修改樣式,本文給大家提到了滾動(dòng)固定位置有多種方法,感興趣的朋友跟隨小編一起看看吧
    2019-05-05

最新評(píng)論