Vue的父子組件通信(十種)

      網友投稿 903 2022-05-30

      面試官:Vue 中父子組件通信有哪些方式?

      自己先想一分鐘。

      無可否認,現在無論大廠還是小廠都已經用上了 Vue.js 框架,簡單易上手不說,教程詳盡,社區活躍,第三方套件還多。真的是前端開發人員必備技能。而且在面試當中也往往會問到關于 Vue 方面的各種問題,其中大部分面試官會問到如上這種問題。

      最近一直在做 Vue項目代碼層面上的優化,說實話,優化別人的代碼真是件痛苦的事情,功能實現尚且不說,就說代碼規范我就能再寫出一篇文章來。真的是無規范不成方圓,規范這個東西太重?要了!有點扯了,回到主題,咳咳,那就談談我對上面的面試題的理解吧,文筆有限,不妥之處,歡迎在文章結尾留言斧正啊,正啊,?。?/p>

      概述

      幾種通信方式無外乎以下幾種:

      Prop(常用)

      面試官:Vue 中父子組件通信有哪些方式?

      自己先想一分鐘。

      無可否認,現在無論大廠還是小廠都已經用上了 Vue.js 框架,簡單易上手不說,教程詳盡,社區活躍,第三方套件還多。真的是前端開發人員必備技能。而且在面試當中也往往會問到關于 Vue 方面的各種問題,其中大部分面試官會問到如上這種問題。

      最近一直在做 Vue項目代碼層面上的優化,說實話,優化別人的代碼真是件痛苦的事情,功能實現尚且不說,就說代碼規范我就能再寫出一篇文章來。真的是無規范不成方圓,規范這個東西太重?要了!有點扯了,回到主題,咳咳,那就談談我對上面的面試題的理解吧,文筆有限,不妥之處,歡迎在文章結尾留言斧正啊,正啊,?。?/p>

      概述

      幾種通信方式無外乎以下幾種:

      Prop(常用)

      $emit?(組件封裝用的較多)

      .sync語法糖 (較少)

      $attrs?和?$listeners?(組件封裝用的較多)

      provide?和?inject?(高階組件/組件庫用的較多)

      其他方式通信

      詳述

      下面逐個介紹,大神請繞行。

      1. Prop

      英式發音:[pr?p]。這個在我們日常開發當中用到的非常多。簡單來說,我們可以通過 Prop 向子組件傳遞數據。用一個形象的比喻來說,父子組件之間的數據傳遞相當于自上而下的下水管子,只能從上往下流,不能逆流。這也正是 Vue 的設計理念之單向數據流。而 Prop 正是管道與管道之間的一個銜接口,這樣水(數據)才能往下流。說這么多,看代碼:

      // Js let Child = Vue.extend({ template: '

      {{ content }}

      ', props: { content: { type: String, default: () => { return 'from child' } } } }) new Vue({ el: '#app', data: { message: 'from parent' }, components: { Child } })

      瀏覽器輸出:

      from parent

      2. $emit

      英式發音:[i?m?t]。官方說法是觸發當前實例上的事件。附加參數都會傳給-回調。按照我的理解不知道能不能給大家說明白,先簡單看下代碼吧:

      let MyButton = Vue.extend({ template: '', data () { return { greeting: 'vue.js!' } }, methods: { triggerClick () { this.$emit('greet', this.greeting) } } }) new Vue({ el: '#app', components: { MyButton }, methods: { sayHi (val) { alert('Hi, ' + val) // 'Hi, vue.js!' } } })

      大致邏輯是醬嬸兒的:當我在頁面上點擊按鈕時,觸發了組件MyButton上的監聽事件greet,并且把參數傳給了回調函數sayHi。說白了,當我們從子組件 Emit(派發) 一個事件之前,其內部都提前在事件隊列中 On(監聽)了這個事件及其監聽回調。其實相當于下面這種寫法:

      Vue的父子組件通信(十種)

      vm.$on('greet', function sayHi (val) { console.log('Hi, ' + val) }) vm.$emit('greet', 'vue.js') // => "Hi, vue.js"

      3. .sync 修飾符

      這個家伙在 vue@1.x 的時候曾作為雙向綁定功能存在,即子組件可以修改父組件中的值。因為它違反了單向數據流的設計理念,所以在 vue@2.0 的時候被干掉了。但是在 vue@2.3.0+ 以上版本又重新引入了這個 .sync 修飾符。但是這次它只是作為一個編譯時的語法糖存在。它會被擴展為一個自動更新父組件屬性的 v-on -。說白了就是讓我們手動進行更新父組件中的值了,從而使數據改動來源更加的明顯。下面引入自官方的一段話:

      在有些情況下,我們可能需要對一個 prop 進行“雙向綁定”。不幸的是,真正的雙向綁定會帶來維護上的問題,因為子組件可以修改父組件,且在父組件和子組件都沒有明顯的改動來源。

      既然作為一個語法糖,肯定是某種寫法的簡寫形式,哪種寫法呢,看代碼:

      于是我們可以用.sync語法糖簡寫成如下形式:

      廢話這么多,如何做到“雙向綁定” 呢?讓我們進段廣告,廣告之后更加精彩! ... 好的,歡迎回來。假如我們想實現這樣一個效果:改變子組件文本框中的值同時改變父組件中的值。怎么做?列位不妨先想想。先看段代碼:

      {{ userName }}
      let Login = Vue.extend({ template: `
      `, props: ['name'], data () { return { text: '' } }, watch: { text (newVal) { this.$emit('update:name', newVal) } } }) new Vue({ el: '#app', data: { userName: '' }, components: { Login } })

      下面劃重點,代碼里有這一句話:

      this.$emit('update:name', newVal)

      官方語法是:update:myPropName其中myPropName表示要更新的 prop 值。當然如果你不用 .sync 語法糖使用上面的 .$emit 也能達到同樣的效果。僅此而已!

      4.$attrs和$listeners

      官網對$attrs的解釋如下:

      包含了父作用域中不作為prop被識別 (且獲取) 的特性綁定 (class和style除外)。當一個組件沒有聲明任何 prop 時,這里會包含所有父作用域的綁定 (class和style除外),并且可以通過v-bind="$attrs"傳入內部組件——在創建高級別的組件時非常有用。

      官網對$listeners的解釋如下:

      包含了父作用域中的 (不含.native修飾器的)v-on事件-。它可以通過v-on="$listeners"傳入內部組件——在創建更高層次的組件時非常有用。

      我覺得$attrs和$listeners屬性像兩個收納箱,一個負責收納屬性,一個負責收納事件,都是以對象的形式來保存數據。看下面的代碼解釋:

      從 Html 中可以看到,這里有倆屬性和倆方法,區別是屬性一個是prop聲明,事件一個是.native修飾器。

      let Child = Vue.extend({ template: '

      {{ foo }}

      ', props: ['foo'], created () { console.log(this.$attrs, this.$listeners) // -> {bar: "parent bar"} // -> {two: fn} // 這里我們訪問父組件中的 `triggerTwo` 方法 this.$listeners.two() // -> 'two' } }) new Vue({ el: '#app', data: { foo: 'parent foo', bar: 'parent bar' }, components: { Child }, methods: { triggerOne () { alert('one') }, triggerTwo () { alert('two') } } }) 可以看到,我們可以通過$attrs和$listeners進行數據傳遞,在需要的地方進行調用和處理,還是很方便的。當然,我們還可以通過v-on="$listeners"一級級的往下傳遞,子子孫孫無窮盡也! 一個插曲! 當我們在組件上賦予了一個非Prop 聲明時,編譯之后的代碼會把這些個屬性都當成原始屬性對待,添加到 html 原生標簽上,看上面的代碼編譯之后的樣子:

      parent foo

      這樣會很難看,同時也爆了某些東西。如何去掉?這正是 inheritAttrs 屬性的用武之地!給組件加上這個屬性就行了,一般是配合$attrs使用??创a: // 源碼 let Child = Vue.extend({ ... inheritAttrs: false, // 默認是 true ... })

      可以看到,我們可以通過$attrs和$listeners進行數據傳遞,在需要的地方進行調用和處理,還是很方便的。當然,我們還可以通過v-on="$listeners"一級級的往下傳遞,子子孫孫無窮盡也!

      一個插曲!

      當我們在組件上賦予了一個非Prop 聲明時,編譯之后的代碼會把這些個屬性都當成原始屬性對待,添加到 html 原生標簽上,看上面的代碼編譯之后的樣子:

      parent foo

      這樣會很難看,同時也爆了某些東西。如何去掉?這正是 inheritAttrs 屬性的用武之地!給組件加上這個屬性就行了,一般是配合$attrs使用。看代碼:

      // 源碼

      let Child = Vue.extend({

      ...

      inheritAttrs: false, // 默認是 true

      ...

      })

      再次編譯:

      parent foo

      5.provide/inject

      他倆是對CP, 感覺挺神秘的。來看下官方對 provide / inject 的描述:

      provide和inject主要為高階插件/組件庫提供用例。并不推薦直接用于應用程序代碼中。并且這對選項需要一起使用,以允許一個祖先組件向其所有子孫后代注入一個依賴,不論組件層次有多深,并在起上下游關系成立的時間里始終生效。

      看完描述有點懵懵懂懂!一句話總結就是:小時候你老爸什么東西都先幫你存著等你長大該娶媳婦兒了你要房子給你買要車給你買只要他有的盡量都會滿足你。下面是這句話的代碼解釋:

      let Son = Vue.extend({ template: '

      son

      ', inject: { house: { default: '沒房' }, car: { default: '沒車' }, money: { // 長大工作了雖然有點錢 // 僅供生活費,需要向父母要 default: '¥4500' } }, created () { console.log(this.house, this.car, this.money) // -> '房子', '車子', '¥10000' } }) new Vue({ el: '#app', provide: { house: '房子', car: '車子', money: '¥10000' }, components: { Son } })

      6. 其他方式通信

      除了以上五種方式外,其實還有:

      EventBus

      思路就是聲明一個全局Vue實例變量EventBus, 把所有的通信數據,事件監聽都存儲到這個變量上。這樣就達到在組件間數據共享了,有點類似于 Vuex。但這種方式只適用于極小的項目,復雜項目還是推薦 Vuex。下面是實現 EventBus 的簡單代碼:

      // 全局變量 let EventBus = new Vue() // 子組件 let Child = Vue.extend({ template: '

      child

      ', created () { console.log(EventBus.message) // -> 'hello' EventBus.$emit('received', 'from child') } }) new Vue({ el: '#app', components: { Child }, created () { // 變量保存 EventBus.message = 'hello' // 事件監聽 EventBus.$on('received', function (val) { console.log('received: '+ val) // -> 'received: from child' }) } })

      Vuex

      官方推薦的,Vuex 是一個專為 Vue.js 應用程序開發的狀態管理模式。

      $parent

      父實例,如果當前實例有的話。通過訪問父實例也能進行數據之間的交互,但極小情況下會直接修改父組件中的數據。

      $root

      當前組件樹的根 Vue 實例。如果當前實例沒有父實例,此實例將會是其自己。通過訪問根組件也能進行數據之間的交互,但極小情況下會直接修改父組件中的數據。

      broadcast / dispatch

      他倆是 vue@1.0 中的方法,分別是事件廣播 和 事件派發。雖然 vue@2.0 里面刪掉了,但可以模擬這兩個方法。可以借鑒 Element 實現。有時候還是非常有用的,比如我們在開發樹形組件的時候等等。

      總結

      啰嗦了這么多,希望看到的同學或多或少有點收獲吧。不對的地方還請留言指正,不勝感激。父子組件間的通信其實有很多種,就看你在哪些情況下去用。不同場景不同對待。前提是你要心中有數才行!通過大神之路還有很遠,只要每天看看社區,看看文檔,寫寫Demo,每天進步一點點,總會有收獲的。

      Vue

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      上一篇:Maven入門教程(一)
      下一篇:部署Tomcat負載均衡
      相關文章
      亚洲AV无码一区二区三区牛牛| 国产成人精品日本亚洲18图 | 亚洲国产精品成人精品小说| 亚洲精品国产精品乱码在线观看| 亚洲熟伦熟女新五十路熟妇| 亚洲国产午夜中文字幕精品黄网站 | 国产成人精品日本亚洲网址| 亚洲免费一级视频| 亚洲国产成人精品无码区在线秒播| 亚洲天堂在线播放| 亚洲影院在线观看| 精品亚洲A∨无码一区二区三区| 亚洲第一福利视频| 久久亚洲精品无码aⅴ大香| 麻豆亚洲AV永久无码精品久久 | 内射干少妇亚洲69XXX| 亚洲福利电影一区二区?| 亚洲精品视频免费在线观看| 亚洲日本香蕉视频| 亚洲精品免费网站| 亚洲国产精品自在自线观看| 亚洲成AV人影片在线观看| 国产在亚洲线视频观看| 亚洲第一区精品观看| 亚洲熟妇少妇任你躁在线观看无码| 久久亚洲高清综合| 久久精品国产亚洲一区二区| 亚洲2022国产成人精品无码区| 亚洲狠狠久久综合一区77777| 久久亚洲sm情趣捆绑调教| 亚洲国产精品午夜电影| 精品国产成人亚洲午夜福利| 亚洲成a人无码亚洲成av无码| 亚洲AV无码成H人在线观看| 久久精品亚洲福利| 亚洲AV无码一区二区乱子伦| 亚洲视频一区二区三区| 2020久久精品亚洲热综合一本| 亚洲色无码国产精品网站可下载| 亚洲aⅴ无码专区在线观看春色 | 亚洲白色白色永久观看|