真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

vue組件間如何進行通信

本篇內(nèi)容介紹了“vue組件間如何進行通信”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

成都創(chuàng)新互聯(lián)公司是一家專業(yè)提供進賢企業(yè)網(wǎng)站建設,專注與網(wǎng)站制作、成都網(wǎng)站設計、HTML5、小程序制作等業(yè)務。10年已為進賢眾多企業(yè)、政府機構(gòu)等服務。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站設計公司優(yōu)惠進行中。

vue是數(shù)據(jù)驅(qū)動視圖更新的框架, 我們平時開發(fā),都會把頁面不同模塊拆分成一個一個vue組件, 所以對于vue來說組件間的數(shù)據(jù)通信非常重要,那么組件之間如何進行數(shù)據(jù)通信的呢?

首先我們需要知道在vue中組件之間存在什么樣的關系, 才更容易理解他們的通信方式。

一般我們分為如下關系:

  • 父子組件之間通信

  • 非父子組件之間通信(兄弟組件、隔代關系組件等)

props / $emit

父組件通過props的方式向子組件傳遞數(shù)據(jù),而通過$emit 子組件可以向父組件通信。

  1. 父組件向子組件傳值







注意:

prop 只可以從上一級組件傳遞到下一級組件(父子組件),即所謂的單向數(shù)據(jù)流。而且 prop 只讀,不可被修改,所有修改都會失效并警告。

  • 第一,不應該在一個子組件內(nèi)部改變 prop,這樣會破壞單向的數(shù)據(jù)綁定,導致數(shù)據(jù)流難以理解。如果有這樣的需要,可以通過 data 屬性接收或使用 computed 屬性進行轉(zhuǎn)換。

  • 第二,如果 props 傳遞的是引用類型(對象或者數(shù)組),在子組件中改變這個對象或數(shù)組,父組件的狀態(tài)會也會做相應的更新,利用這一點就能夠?qū)崿F(xiàn)父子組件數(shù)據(jù)的“雙向綁定”,雖然這樣實現(xiàn)能夠節(jié)省代碼,但會犧牲數(shù)據(jù)流向的簡潔性,令人難以理解,最好不要這樣去做。

  • 想要實現(xiàn)父子組件的數(shù)據(jù)“雙向綁定”,可以使用 v-model.sync。

  1. 子組件向父組件傳值

使用 $emit 向父組件傳數(shù)據(jù),原理這樣的: 父組件在子組件通過v-on監(jiān)聽函數(shù)并接收參數(shù),vue框架就在子組件監(jiān)聽了你v-on="fn"的fn事件函數(shù),在子組件使用$emit就能觸發(fā)了,下面寫個例子。







v-model 指令

v-model 是用來在表單控件或者組件上創(chuàng)建雙向綁定的,他的本質(zhì)是 v-bindv-on語法糖,在一個組件上使用 v-model,默認會為組件綁定名為 value 的 prop 和名為 input 的事件。

當我們組件中的某一個 prop 需要實現(xiàn)上面所說的”雙向綁定“時,v-model 就能大顯身手了。有了它,就不需要自己手動在組件上綁定監(jiān)聽當前實例上的自定義事件,會使代碼更簡潔。

下面以一個 input 組件實現(xiàn)的核心代碼,介紹下 v-model 的應用。





上面例子看到,v-model="inputValue" 他的本質(zhì)就是 v-bind 和 v-on 的語法糖,默認為父組件綁定名為 :value="inputValue"的屬性,和@input="(v) => { this.inputValue = v }"事件,子組件通過 this.$emit('input', value) 通知父組件

所以他原理也是利用了我們上面講的父子組件傳參 props / $emit 方式來實現(xiàn)雙向綁定

有時,在某些特定的控件中名為 value 的屬性會有特殊的含義,這時可以通過 v-model 選項來回避這種沖突。

.sync 修飾符

  • .sync 修飾符在 vue 1.x 的版本中就已經(jīng)提供,1.x 版本中,當子組件改變了一個帶有 .syncprop 的值時,會將這個值同步到父組件中的值。這樣使用起來十分方便,但問題也十分明顯,這樣破壞了單向數(shù)據(jù)流,當應用復雜時,debug 的成本會非常高。

  • 于是乎,在vue 2.0中移除了 .sync。但是在實際的應用中,.sync 是有它的應用場景的,所以在 vue 2.3 版本中,又迎來了全新的 .sync。

  • 新的 .sync 修飾符所實現(xiàn)的已經(jīng)不再是真正的雙向綁定,它的本質(zhì)和 v-model 類似,只是一種縮寫。

正常封裝組件例子:

上面的代碼,使用 .sync 就可以寫成

這樣,在子組件中,就可以通過下面代碼來實現(xiàn)對這個 prop 重新賦值了。

this.$emit('update:title', newTitle)

看到這里,是不是發(fā)現(xiàn) .sync 修飾符 和 v-model 很相似,也是語法糖, v-bind:title.sync 也就是 等效于 v-bind:title="doc.title" v-on:update:title="doc.title = $event"

v-model 和 .sync 對比

.sync 從功能上看和 v-model 十分相似,都是為了實現(xiàn)數(shù)據(jù)的“雙向綁定”,本質(zhì)上,也都不是真正的雙向綁定,而是語法糖。

相比較之下,.sync 更加靈活,它可以給多個 prop 使用,而 v-model 在一個組件中只能有一個。

從語義上來看,v-model 綁定的值是指這個組件的綁定值,比如 input 組件select 組件,日期時間選擇組件顏色選擇器組件,這些組件所綁定的值使用 v-model 比較合適。其他情況,沒有這種語義,個人認為使用 .sync 更好。

parent/parent /parent/children

通過$parent$children就可以訪問組件的實例,拿到實例代表什么?代表可以訪問此組件的所有方法data。列子如下:







要注意邊界情況,如在#app上拿$parent得到的是new Vue()的實例,在這實例上再拿$parent得到的是undefined,而在最底層的子組件拿$children是個空數(shù)組。也要注意得到parentparent和parent和children的值不一樣,$children 的值是數(shù)組,而$parent是個對象

props $emit 、 $parent $children兩種方式用于父子組件之間的通信, 而使用props進行父子組件通信更加普遍,二者皆不能用于非父子組件之間的通信。

provide / inject

provide / inject 是vue2.2.0新增的api, 簡單來說就是父組件中通過provide來提供變量, 然后再子組件中通過inject來注入變量。

官方描述: 這對選項需要一起使用,以允許一個祖先組件向其所有子孫后代注入一個依賴,不論組件層次有多深,并在其上下游關系成立的時間里始終生效

provide 選項應該是

  • 一個對象或返回一個對象的函數(shù)。該對象包含可注入其子孫的屬性。在該對象中你可以使用 ES2015 Symbols 作為 key,但是只在原生支持 Symbol 和 Reflect.ownKeys 的環(huán)境下可工作。

inject 選項應該是:

  • 一個字符串數(shù)組

  • 一個對象(詳情點擊這里)

基本用法

// 祖先組件 提供foo
//第一種
export default {
  name: "father",
  provide() {
    return {
      foo: 'hello'
    }
  },
}
//第二種
export default {
  name: "father",
  provide: {
    foo:'hello~~~~'
  },
}
//后代組件 注入foo, 直接當做this.foo來用
export default {
  inject:['foo'],
}

上面的兩種用法有什么區(qū)別嗎?

  • 如果你只是傳一個字符串,像上面的hello,那么是沒有區(qū)別的,后代都能讀到。

  • 如果你需要this對象屬性的值(如下所示代碼),那么第二種是傳不了的,后代組件拿不到數(shù)據(jù)。所以建議只寫第一種

//當你傳遞對象給后代時
provide() {
    return {
        test: this.msg
    }
},

注意:一旦注入了某個數(shù)據(jù),比如上面示例中的 foo,那這個組件中就不能再聲明 foo 這個數(shù)據(jù)了,因為它已經(jīng)被父級占有。

provide 和 inject 綁定并不是可響應的。

這是刻意為之的。然而,如果你傳入了一個可監(jiān)聽的對象,那么其對象的屬性還是可響應的。因為對象是引用類型。

先來個值類型的數(shù)據(jù)(也就是字符串)例子,不會響應

provide(){
  return{
    test:this.msg
  }
},
data() {
  return {
    msg: "Welcome to Your Vue.js App",
  }
}
mounted(){
  setTimeout(()=>{
    this.msg = "halo world";
    console.log(this._provided.msg)
    //log:Welcome to Your Vue.js App
  },3000)
},

如上所示,這樣做是不行的,打印出來的 _provided 中的數(shù)據(jù)并沒有改,子組件取得值也沒變。

你甚至可以直接給 this._provided.msg 賦值,但是即使是_provided.msg 里面的值改變了,子組件的取值,依然沒有變。

當你的參數(shù)是對象的時候,就可以響應了,如下:

provide(){
  return{
    test:this.activeData
  }
},
data() {
  return {
    activeData:{name:'halo'},
  }
}
mounted(){
  setTimeout(()=>{
    this.activeData.name = 'world';
  },3000)
}

這就是vue官方中寫道的對象的屬性是可以響應的

provide/inject 實現(xiàn)全局變量

provide/inject不是只能從祖先傳遞給后代嗎?是的,但是,如果我們綁定到最頂層的組件app.vue,是不是所有后代都接收到了,就是當做全局變量來用了。

//app.vue
export default {
  name: 'App',
  provide(){
    return{
      app:this
    }
  },
  data(){
    return{
      text:"it's hard to tell the night time from the day"
    }
  },
  methods:{
    say(){
      console.log("Desperado, why don't you come to your senses?")
    }
  }
}
//其他所有子組件,需要全局變量的,只需要按需注入app即可
export default {
  inject:['foo','app'],
  mounted(){
    console.log(this.app.text); // 獲取app中的變量
    this.app.say(); // 可以執(zhí)行app中的方法,變身為全局方法!
  }
}

provide/inject 實現(xiàn)頁面刷新,不閃爍

  1. vue-router重新路由到當前頁面,頁面是不進行刷新的

  2. 采用window.reload(),或者router.go(0)刷新時,整個瀏覽器進行了重新加載,閃爍,體驗不好

那我們怎么做呢?

跟上面的原理差不多,我們只在控制路由的組件中寫一個函數(shù)(使用v-if控制router-view的顯示隱藏,這里的原理不作贅述),然后把這個函數(shù)傳遞給后代,然后在后代組件中調(diào)用這個方法即可刷新路由啦。

//app.vue


export default {
  name: 'App',
  provide() {
    return {
      reload: this.reload
    }
  },
  data() {
    return {
      isShowRouter: true,
    }
  },
  methods:{
    reload() {
      this.isShowRouter = false;
      this.$nextTick(() => { 
        this.isShowRouter = true;
      })
    }
  }
}
//后代組件
export default {
  inject: ['reload'],  
}

這里 provide 使用了函數(shù)傳遞給后代,然后后代調(diào)用這個函數(shù),這種思路,也是可以做子后代向父組件傳參通訊的思路了。這里的原理,和 event 事件訂閱發(fā)布就很像了

ref / $refs

ref:如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子組件上,引用就指向組件實例,可以通過實例直接調(diào)用組件的方法或訪問數(shù)據(jù), 我們看一個ref 來訪問組件的例子:

// 子組件 A.vue

export default {
  data () {
    return {
      name: 'Vue.js'
    }
  },
  methods: {
    sayHello () {
      console.log('hello')
    }
  }
}
// 父組件 app.vue


ref 這種方式,就是獲取子組件的實例,然后可以直接子組件的方法和訪問操作data的數(shù)據(jù),就是父組件控制子組件的一種方式,子組件想向父組件傳參或操作,只能通過其他的方式了

eventBus

eventBus呢,其實原理就是 事件訂閱發(fā)布,eventBus 又稱為事件總線,在vue中可以使用它來作為溝通橋梁的概念, 就像是所有組件共用相同的事件中心,可以向該中心注冊發(fā)送事件或接收事件, 所以組件都可以通知其他組件。

這里我們可以直接使用 vue 自帶的事件監(jiān)聽,也就是 emitemitemiton,我們來簡單封裝下:

  1. 首先需要創(chuàng)建一個事件總線并將其導出, 以便其他模塊可以使用或者監(jiān)聽它.

新建一個 event-bus.js 文件

// event-bus.js

import Vue from 'vue'
export const EventBus = new Vue()
  1. 發(fā)生事件

假設你有兩個組件: additionNum 和 showNum, 這兩個組件可以是兄弟組件也可以是父子組件;這里我們以兄弟組件為例:



// addtionNum.vue 中發(fā)送事件



  1. 接收事件

// showNum.vue 中接收事件



這樣就實現(xiàn)了在組件addtionNum.vue中點擊相加按鈕, 在showNum.vue中利用傳遞來的 num 展示求和的結(jié)果.

  1. 移除事件監(jiān)聽者

如果想移除事件的監(jiān)聽, 可以像下面這樣操作:

import { eventBus } from 'event-bus.js'
EventBus.$off('addition')

自己封裝一套 eventBus

這里使用自己封裝一套eventBus也行,方便自己想干啥就干啥, 下面貼封裝好的一套給大家

/* eslint-disable no-console */
// 事件映射表
let eventMap = {}

/**
 * 監(jiān)聽事件
 * @param {string}    eventName 事件名
 * @param {function}  listener 回調(diào)函數(shù)
 * @param {object}    instance 注冊事件的實例
 */
function on(eventName, listener, instance) {
  eventMap[eventName] = eventMap[eventName] || []
  eventMap[eventName].push({
    listener,
    instance,
  })
}

// 監(jiān)聽事件,只執(zhí)行一次
function once(eventName, listener, instance) {
  eventMap[eventName] = eventMap[eventName] || []
  eventMap[eventName].push({
    listener,
    instance,
    once: true,
  })
}

// 解除事件監(jiān)聽
function off(eventName, listener) {
  // 解除所有事件監(jiān)聽
  if (!eventName) {
    eventMap = {}
    return
  }

  // 沒有對應事件
  if (!eventMap[eventName]) {
    return
  }

  // 解除某事件監(jiān)聽
  eventMap[eventName].forEach((currentEvent, index) => {
    if (currentEvent.listener === listener) {
      eventMap[eventName].splice(index, 1)
    }
  })
}

// 發(fā)送事件,執(zhí)行對應響應函數(shù)
function emit(eventName, ...args) {
  if (!eventMap[eventName]) {
    return
  }

  eventMap[eventName].forEach((currentEvent, index) => {
    currentEvent.listener(...args)
    if (currentEvent.once) {
      eventMap[eventName].splice(index, 1)
    }
  })
}

// 顯示當前注冊的事件,代碼優(yōu)化時使用
function showEventMap(targetEventName) {
  if (targetEventName) { // 查看具體某個事件的監(jiān)聽情況
    eventMap[targetEventName].forEach(eventItem => {
      console.log(targetEventName, eventItem.instance, eventItem.listener)
    })
  } else { // 查看所以事件的監(jiān)聽情況
    Object.keys(eventMap).forEach(eventName => {
      eventMap[eventName].forEach(eventItem => {
        console.log(eventName, eventItem.instance, eventItem.listener)
      })
    })
  }
}

// 提供 vue mixin 方法,在 beforeDestroy 自動注銷事件監(jiān)聽
export const mixin = {
  created() {
    // 重載 on 函數(shù),收集本組件監(jiān)聽的事件,待消除時,銷毀事件監(jiān)聽
    this.$eventListenerList = []
    this.$event = { off, once, emit, showEventMap }
    this.$event.on = (eventName, listener) => {
      this.$eventListenerList.push({ eventName, listener })
      on(eventName, listener)
    }
  },

  // 消除組件時,自動銷毀事件監(jiān)聽
  beforeDestroy() {
    this.$eventListenerList.forEach(currentEvent => {
      off(currentEvent.eventName, currentEvent.listener)
    })
  },
}

export default { on, off, once, emit, showEventMap }

如何使用呢,只需在 項目的 main.js, 引入 ,然后 Vue.mixin 即可,如下:

// main.js
import Vue from 'vue'
import { mixin as eventMixin } from '@/event/index'

Vue.mixin(eventMixin)

在vue項目其他文件,就可以直接 this.$event.on this.$event.$emit  如下:

this.$event.on('test', (v) => { console.log(v) })   this.$event.$emit('test', 1)

還順便封裝了個mixin, 好處呢,就是在vue頁面監(jiān)聽事件后,頁面銷毀后,也自動銷毀了事件監(jiān)聽

Vuex

Vuex介紹

  • Vuex 是一個專為 Vue.js 應用程序開發(fā)的狀態(tài)管理模式。它采用集中式存儲管理應用的所有組件的狀態(tài),并以相應的規(guī)則保證狀態(tài)以一種可預測的方式發(fā)生變化.

  • Vuex 解決了多個視圖依賴于同一狀態(tài)和來自不同視圖的行為需要變更同一狀態(tài)的問題,將開發(fā)者的精力聚焦于數(shù)據(jù)的更新而不是數(shù)據(jù)在組件之間的傳遞上

Vuex各個模塊

  • state:用于數(shù)據(jù)的存儲,是store中的唯一數(shù)據(jù)源

  • getters:如vue中的計算屬性一樣,基于state數(shù)據(jù)的二次包裝,常用于數(shù)據(jù)的篩選和多個數(shù)據(jù)的相關性計算

  • mutations:類似函數(shù),改變state數(shù)據(jù)的唯一途徑,且不能用于處理異步事件

  • actions:類似于mutation,用于提交mutation來改變狀態(tài),而不直接變更狀態(tài),可以包含任意異步操作

  • modules:類似于命名空間,用于項目中將各個模塊的狀態(tài)分開定義和操作,便于維護

Vuex實例應用

這里我們先新建 store文件夾, 對Vuex進行一些封裝處理

在 store 文件夾下添加 index.js 文件

// index.js

// 自動掛載指定目錄下的store
import Vue from 'vue'
import Vuex from 'vuex'

Vue.use(Vuex)

let modules = {}

// @/store/module 目錄下的文件自動掛載為 store 模塊
const subModuleList = require.context('@/store/modules', false, /.js$/)
subModuleList.keys().forEach(subRouter => {
  const moduleName = subRouter.substring(2, subRouter.length - 3)
  modules[moduleName] = subModuleList(subRouter).default
})

export default new Vuex.Store({
  state: {},
  mutations: {},
  actions: {},
  modules
})

在 store 文件夾下添加 module 文件夾,在module文件夾再新建 user.js 文件

// user.js

import user from '@/utils/user.js'
import userApi from '@/apis/user'
import { OPEN_ACCOUNT_STAGE, STAGE_STATUS } from '@/constant'

let getUserPromise = null

export default {
  namespaced: true,
  state() {
    return {
      userInfo: null, // 用戶信息
      isLogined: !!user.getToken(), // 是否已經(jīng)登錄
    }
  },
  mutations: {
    // 更新用戶信息
    updateUser(state, payload) {
      state.isLogined = !!payload
      state.userInfo = payload
    },
  },
  actions: {
    // 獲取當前用戶信息
    async getUserInfo(context, payload) {
      // forceUpdate 表示是否強制更新
      if (context.state.userInfo && !payload?.forceUpdate) {
        return context.state.userInfo
      }
      if (!getUserPromise || payload?.forceUpdate) {
        getUserPromise = userApi.getUserInfo()
      }
      // 獲取用戶信息
      try {
        const userInfo = await getUserPromise
        context.commit('updateUser', userInfo)
      } finally {
        getUserPromise = null
      }
      return context.state.userInfo
    },

    // 登出
    async logout(context, payload = {}) {
      // 是否手動退出
      const { manual } = payload
      if (manual) {
        await userApi.postLogout()
      }
      user.clearToken()
      context.commit('updateUser', null)
    },
  }
}

然后在項目的 main.js 文件中引入

import Vue from 'vue'
import App from '@/app.vue'
import { router } from '@/router'
import store from '@/store/index'

const vue = new Vue({
  el: '#app',
  name: 'root',
  router,
  store,
  render: h => h(App),
})

封裝的很愉快了,然后就正常操作即可。

this.$store.state.user.isLogined
this.$store.state.user.userInfo
this.$store.commit('user/updateUser', {})
 await this.$store.dispatch('user/logout', { manual: true })

localStorage / sessionStorage

這種通信比較簡單,缺點是數(shù)據(jù)和狀態(tài)比較混亂,不太容易維護。

  • 通過window.localStorage.getItem(key)獲取數(shù)據(jù)

  • 通過window.localStorage.setItem(key,value)存儲數(shù)據(jù)

注意用JSON.parse() / JSON.stringify() 做數(shù)據(jù)格式轉(zhuǎn)換, localStorage / sessionStorage可以結(jié)合vuex, 實現(xiàn)數(shù)據(jù)的持久保存,同時使用vuex解決數(shù)據(jù)和狀態(tài)混亂問題.

自己實現(xiàn)簡單的 Store 模式

對于小型的項目,通信十分簡單,這時使用 Vuex 反而會顯得冗余和繁瑣,這種情況最好不要使用 Vuex,可以自己在項目中實現(xiàn)簡單的 Store。

// store.js
const store = {
  debug: true,
  state: {
    author: 'yushihu!'
  },
  setAuthorAction (newValue) {
    if (this.debug) console.log('setAuthorAction triggered with', newValue)
    this.state.author = newValue
  },
  deleteAuthorAction () {
    if (this.debug) console.log('deleteAuthorAction triggered')
    this.state.author = ''
  }
}
export default store

上面代碼原理就是,store.js文件暴露出一個對象 store,通過引入 store.js 文件 各個頁面來共同維護這個store對象

和 Vuex 一樣,store 中 state 的改變都由 store 內(nèi)部的 action 來觸發(fā),并且能夠通過 console.log() 打印觸發(fā)的痕跡。這種方式十分適合在不需要使用 Vuex 的小項目中應用。

$root 訪問根實例的方法相比,這種集中式狀態(tài)管理的方式能夠在調(diào)試過程中,通過 console.log() 記錄來確定當前變化是如何觸發(fā)的,更容易定位問題。

通過 $root 訪問根實例

通過 $root,任何組件都可以獲取當前組件樹的根 Vue 實例,通過維護根實例上的 data,就可以實現(xiàn)組件間的數(shù)據(jù)共享。

//main.js 根實例
new Vue({
    el: '#app',
    store,
    router,
    // 根實例的 data 屬性,維護通用的數(shù)據(jù)
    data: function () {
        return {
            author: ''
        }
    },
    components: { App },
    template: '',
});







注意:通過這種方式,雖然可以實現(xiàn)通信,但在應用的任何部分,任何時間發(fā)生的任何數(shù)據(jù)變化,都不會留下變更的記錄,這對于稍復雜的應用來說,調(diào)試是致命的,不建議在實際應用中使用。

attrsattrs與attrs與listeners

現(xiàn)在我們來討論一種情況, 我們一開始給出的組件關系圖中A組件與D組件是隔代關系, 那它們之前進行通信有哪些方式呢?

  1. 使用props綁定來進行一級一級的信息傳遞, 如果D組件中狀態(tài)改變需要傳遞數(shù)據(jù)給A, 使用事件系統(tǒng)一級級往上傳遞

  2. 使用eventBus,這種情況下還是比較適合使用, 但是碰到多人合作開發(fā)時, 代碼維護性較低, 可讀性也低

  3. 使用Vuex來進行數(shù)據(jù)管理, 但是如果僅僅是傳遞數(shù)據(jù), 而不做中間處理,使用Vuex處理感覺有點大材小用了.

所以就有了 $attrs / $listeners ,通常配合 inheritAttrs 一起使用。

inheritAttrs

默認情況下父作用域的不被認作 propsattribute 綁定 (attribute bindings) 將會“回退”且作為普通的 HTML attribute 應用在子組件的根元素上。當撰寫包裹一個目標元素或另一個組件的組件時,這可能不會總是符合預期行為。

通過設置 inheritAttrsfalse,這些默認行為將會被去掉。而通過 (同樣是 2.4 新增的) 實例 property $attrs 可以讓這些 attribute 生效,且可以通過 v-bind 顯性的綁定到非根元素上。

注意:這個選項不影響 classstyle 綁定。

上面是官方描述:還是很難懂。

簡單的說就是

  • inheritAttrs:true 時繼承除props之外的所有屬性

  • inheritAttrs:false 只繼承class 和 style屬性

$attrs:包含了父作用域中不被認為 (且不預期為) props 的特性綁定 (class 和 style 除外),并且可以通過 v-bind="$attrs" 傳入內(nèi)部組件。當一個組件沒有聲明任何 props 時,它包含所有父作用域的綁定 (class 和 style 除外)。

$listeners:包含了父作用域中的 (不含 .native 修飾符) v-on 事件監(jiān)聽器。它可以通過 v-on="$listeners" 傳入內(nèi)部組件。它是一個對象,里面包含了作用在這個組件上的所有事件監(jiān)聽器,相當于子組件繼承了父組件的事件。

講了這么多文字概念,我們還是來看代碼例子吧:

新建一個 father.vue 組件


child.vue 組件:


grandSon.vue 組件


這種方式的傳值雖然說不常用,感覺可讀性不是很好。但其對于組件層級嵌套比較深,使用props會很繁瑣,或者項目比較小,不太適合使用 Vuex 的時候,可以考慮用它

“vue組件間如何進行通信”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!


當前文章:vue組件間如何進行通信
文章出自:http://weahome.cn/article/giejhp.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部