Vue组件通信

总体来说,Vue中组件之间的通信场景如下图:

组件通信场景

可以将其分为父子组件通信兄弟组件通信跨级组件通信

1. 自定义事件

子组件-->父组件: 采用自定义事件,子组件用$emit()来触发事件,父组件用$on()来监听子组件事件,父组件也可以直接在子组件的自定义标签上使用v-on来监听子组件触发的自定义事件:

<!-- 自定义事件 -->
<body>
    <div id="app">
        <p>SUM: {{total}} </p>
        <!-- 父组件用v-on监听子组件事件,做出处理 -->
        <my-component 
            @increase = "handleGetTotal"
            @reduce = "handleGetTotal"
        ></my-component>
    </div>

    <script src = "https://unpkg.com/vue/dist/vue.min.js"></script>
    <script>
        Vue.component('my-component',{
            template: '<div><button @click="handleIncrease">+1</button> <button @click="handleReduce">-1</button></div>',
            data: function() {
                return {
                    counter: 0
                }
            },
            methods: {
                handleIncrease: function() {
                    this.counter++;
                    //使用$emit触发increase事件,第一个参数为自定义事件的名称,第二个参数是要传递的数据
                    this.$emit('increase', this.counter);
                },
                handleReduce: function() {
                    this.counter--;
                    this.$emit('reduce', this.counter);
                }
            }
        });

        var app = new Vue({
            el: '#app',
            data: {
               total: 0,
            },
            methods: {
                handleGetTotal: function(total) {
                    this.total = total;
                }
            }
        })
    </script>
</body>  
执行结果

子组件有两个按钮,分别实现+1和-1的效果,在改变counter后,通过$emit()把它传递给父组件,父组件通过v-on获取并作出处理。

2. 使用v-model

Vue2.x 中也可以在自定义组件中使用v-model指令:

<!-- 自定义事件使用v-model -->
<body>
    <div id="app">
        <p>SUM: {{total}} </p>
        <my-component v-model="total"></my-component>
    </div>

    <script src = "https://unpkg.com/vue/dist/vue.min.js"></script>
    <script>
        Vue.component('my-component',{
            template: '<div><button @click="handleIncrease">+1</button><button @click="handleReduce">-1</button></div>',
            data: function() {
                return {
                    counter: 0
                }
            },
            methods: {
                handleIncrease: function() {
                    this.counter++;
                    this.$emit('input', this.counter);
                },
                handleReduce: function() {
                    this.counter--;
                    this.$emit('input', this.counter);
                }
            }
        });

        var app = new Vue({
            el: '#app',
            data: {
               total: 0,
            },
        })
    </script>
</body>  

所实现效果一样,但是现在子组件$emit()的事件名为特殊的input,在父组件中,直接使用了v-model绑定一个数据total。这其实是一个语法糖,其实际实现为:

<my-component @input="handleGetTatal"></my-component>
...
methods: {
    handleGetTotal: function(total) {
        this.total = total;
    }
}

v-model还可以用来创建自定义的表单输入组件,进行数据双向绑定。

3. 非父子组件通信--中央事件总线

在Vue 2.x中,推荐使用一个空的Vue实例作为中央事件总线(bus),直接看代码理解:

<!-- 非父子组件通信--中央事件总线 -->
<body>
    <div id="app">
        <p>{{message}}</p>
        <my-component></my-component>
    </div>

    <script src = "https://unpkg.com/vue/dist/vue.min.js"></script>
    <script>
        var bus = new Vue();

        Vue.component('my-component',{
            template: '<div><button @click="handleEvent"">传递事件</button></div>',
            
            methods: {
                handleEvent: function() {
                    bus.$emit('on-message', '来自组件my-component的内容');
                },
            }
        });

        var app = new Vue({
            el: '#app',
            data: {
               message: '',
            },
            mounted: function() {
                var _this = this;
                //在实例初始化时,监听来自bus实例的事件  
                bus.$on('on-message', function(msg) {
                    _this.message = msg;
                });
            }
        })
    </script>
</body>

上述代码中为了方便起见,我们还是以父子组件为例的,但目的是为了理解代码中名为bus的空Vue实例。

bus实例中,没有任何内容,实例app初始化时,会在生命周期钩子函数mounted中监听来自bus的事件on-message,而组件my-component中的按钮点击将on-message事件通过bus传递了给了app。

如果深入使用,可以扩展bus实例,为他添加data、methods、computed等选项,这些都是公用的,在实际业务中,比如用户登录的昵称、性别、邮箱等通用信息,只需要在初始化时被bus获取一次,则之后其他组件就可以直接使用了。

4. 父链 & 子组件索引

除了中央事件总线bus外,还有两种方法可以实现组件之间的通信,父链子组件索引

**父链 **

在子组件中,使用this.$parent可以直接访问该组件的父实例,父组件也可以通过this.$children访问它的所有子组件,从而完成递归向上或向下的访问:

<!-- 父链 -->
<body>
    <div id="app">
        <p>{{message}}</p>
        <my-component></my-component>
    </div>

    <script src = "https://unpkg.com/vue/dist/vue.min.js"></script>
    <script>
        Vue.component('my-component',{
            template: '<div><button @click="handleEvent"">通过父链直接修改父组件数据</button></div>',
            
            methods: {
                handleEvent: function() {
                    //访问到父链并修改其数据
                    this.$parent.message = '来自子组件的修改数据';
                },
            }
        });

        var app = new Vue({
            el: '#app',
            data: {
               message: '我本来的数据',
            },
        })
    </script>
</body>  
执行结果

在业务中,应该尽可能避免子组件依赖父组件中的数据,更不应该主动修改它的数据,这不满足父子组件之间解耦的原则。

理想情况下,只有组件自己可以修改自己的状态。父子组件最好还是通过props$emit来完成通信。

子组件索引

当子组件较多时,通过this.$children来遍历出所需要的子组件时比较困难的,因此Vue提供了子组件索引的方法,用特殊的属性ref来为子组件指定一个索引名称:

<!-- 子组件索引 -->
<body>
    <div id="app">
        <div>{{message}}</div>
        <button @click="handleRef">通过ref获取子组件实例</button>
        <my-component ref="myComponent"></my-component>
    </div>

    <script src = "https://unpkg.com/vue/dist/vue.min.js"></script>
    <script>
        Vue.component('my-component',{
            data: function() {
                return {
                    message: '子组件内容'
                }
            }
        });

        var app = new Vue({
            el: '#app',
            data: function() {
                return {
                    message: ''
                }
            },

            methods: {
                handleRef: function() {
                    //通过$refs访问指定子组件实例
                    var msg = this.$refs.myComponent.message;
                    this.message = msg;
                }
            }
        })
    </script>
</body>  

上述代码,子组件标签上使用ref指定一个名称,并在父组件内通过this.$refs来访问到相应子组件。

$refs只在组件渲染完成后才填充,并且它时非响应式的,仅仅作为直接访问子组件的应急方案,应尽量避免使用。

参考

  1. 《Vue.js 实战》
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,002评论 6 509
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,777评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,341评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,085评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,110评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,868评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,528评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,422评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,938评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,067评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,199评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,877评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,540评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,079评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,192评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,514评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,190评论 2 357

推荐阅读更多精彩内容

  • 能工摹形,巧匠窃意。必三省吾身,万不可怠惰因循。 foreword 这篇容纳了我个人所知道的一些Vue 2.x组件...
    禾小沐的技术与生活阅读 376评论 0 1
  • 父子组件通信 1、父子组件通过prop传递数据 父组件可以将一条数据传递给子组件,这条数据可以是动态的,父组件的数...
    视觉派Pie阅读 1,267评论 0 18
  • Vue组件通信 Vue组件关系可分为三大类: 父子组件 兄弟组件 跨级组件, 相应的组件之间的通信也分类三大类: ...
    dino小恐龙阅读 1,908评论 0 2
  • 背景   Vue是单页面应用,单页面应用又是由组件构成,各个组件之间又互相关联,那么如何实现组件之间通信就显得尤为...
    A郑家庆阅读 897评论 0 1
  • 十九血恋 生与死有时不是人类能够控制,何况疾病的来临并没有太多的先兆,缘海回家后知道了父亲的病重,父亲是缘海生命中...
    易写发阅读 190评论 0 1