vue.js 组件之间传递数据

前言
组件是 vue.js 最强大的功能之一,而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用。如何传递数据也成了组件的重要知识点之一。

组件
组件与组件之间,还存在着不同的关系。父子关系与兄弟关系(不是父子的都暂称为兄弟吧)。

父子组件
父子关系即是组件 A 在它的模板中使用了组件 B,那么组件 A 就是父组件,组件 B 就是子组件。

// 注册一个子组件
Vue.component('child', { 
     data: function(){
          text: '我是father的子组件!' 
     } ,
    template: '<span>{{ text }}</span>'})
// 注册一个父组件
Vue.component('father', { 
     template: '<div><child></child></div>' 
// 在模板中使用了child组件
})

直接使用 father 组件的时候:
<div id="app"> <father></father></div>
页面中就会渲染出 :我是father的子组件!
father 组件在模板中使用了 child 组件,所以它就是父组件,child 组件被使用,所以 child 组件就是子组件。

兄弟组件
两个组件互不引用,则为兄弟组件。

Vue.component('brother1', { 
     template: '<div>我是大哥</div>'
})
Vue.component('brother2', {
     template: '<div>我是小弟</div>'
})

使用组件的时候:
<div id="app"> <brother1></brother1> <brother2></brother2></div>

页面中就会渲染出 : 我是大哥 我是小弟

Prop
子组件想要使用父组件的数据,我们需要通过子组件的 props 选项来获得父组件传过来的数据。以下我使用在 .vue 文件中的格式来写例子。
如何传递数据
在父组件 father.vue 中引用子组件 child.vue,把 name 的值传给 child 组件。

<template> 
    <div class="app"> // message 定义在子组件的 props 中
         <child :message="name"></child> </div>
    </template>
<script> 
    import child from './child.vue';
    export default { 
        components: { child }, 
        data() { 
            return { 
                name: 'linxin' 
            } 
        }
 }
</script>

在子组件 child.vue 中的 props 选项中声明它期待获得的数据

<template>
       <span>Hello {{message}}</span>
</template>
<script> 
       export default { 
         // 在 props 中声明获取父组件的数据通过 message 传过来 
       props:['message'] }
</script>

那么页面中就会渲染出:Hello linxin

单向数据流
当父组件的 name 发生改变,子组件也会自动地更新视图。但是在子组件中,我们不要去修改 prop。如果你必须要修改到这些数据,你可以使用以下方法:
方法一:把 prop 赋值给一个局部变量,然后需要修改的话就修改这个局部变量,而不影响 prop

export default { 
      data(){ 
          newMessage: null
       },
      props: ['message'], 
      created(){ 
            this.newMessage = this.message;
      }
}

方法二:在计算属性中对 prop 进行处理

export default {
     props: ['message'],
     computed{ 
         newMessage(){ 
             return this.newMessage + ' 哈哈哈';
          }
     }
}

自定义事件
prop 是单向绑定的:当父组件的属性变化时,将传导给子组件,但是不会反过来。修改子组件的 prop 值,是不会传回给父组件去更新视图的。那么子组件要如何去与父组件通讯呢?
那就是自定义事件。通过在父组件 $on(eventName) 监听自定义事件,当子组件里 $emit(eventName) 触发该自定义事件的时候,父组件执行相应的操作。
比如在父组件中控制一个弹框子组件的显示,在子组件中按下关闭之后,告诉父组件去隐藏它,然后父组件就执行操作隐藏弹框。

<template>
     <div class="app"> 
// hide 为自定义事件,名字可以自己随便起,不能有大写字母,可以使用短横线 
// @hide 监听子组件触发 hide 事件,则会执行 hideDialog 方法
         <dialog :is-show="show" @hide="hideDialog"></dialog> 
         <button @click="showDialog">显示弹框</button>
     </div>
</template>
<script> 
        import dialog from './dialog.vue'; 
        export default { 
            components: { dialog },
            data() { 
                   return {
                       show: false
                   } 
             }, 
            methods: {
                showDialog() { 
                     this.show = true;
                 }, 
                hideDialog() { 
                     this.show = false; 
                }
          }
 }
</script>

在子组件 dialog.vue 中:

<template>
    <div class="dialog" v-show="isShow"> 
          <p>这里是弹框子组件</p> 
          <button @click="toHide">关闭弹框</button> 
    </div>
</template>
<script> 
    export default {
          // 驼峰式命名的 prop 需要转换为相对应的短横线隔开式 is-show  
          props: ['isShow'],
          methods: { 
               toHide(){
                    // $emit 方法触发父组件的监听事件
                   this.$emit('hide');
                } 
          }
}
</script>

这样就实现了父子组件之间的相互通讯。

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

推荐阅读更多精彩内容