> 웹 프론트엔드 > JS 튜토리얼 > Vue 구성요소와 통신하는 6가지 방법

Vue 구성요소와 통신하는 6가지 방법

hzc
풀어 주다: 2020-06-18 09:44:07
앞으로
3301명이 탐색했습니다.

일반적인 개발 과정에서 아버지와 아들/형제 구성 요소 간의 통신이 반드시 발생하므로 다음은 6개의 Vue 구성 요소 / $e$emit / Vuex$attrs / $listeners

  1. 의 통신 소품에 대한 요약입니다.

    $parent / $children and ref

  2. provide / inject

Foreword

Vue 구성요소와 통신하는 6가지 방법

위 그림과 같이 A/B, B/C, B/D 컴포넌트는 부모-자식입니다. C/D는 형제관계이다. 그렇다면 다양한 사용 시나리오에 따라 다양한 통신 방법을 선택하는 방법은 무엇입니까? 따라서 전제는 다양한 의사소통 방법의 기능과 차이점을 이해해야 한다는 것입니다.

1.props / $emit

이것은 우리가 일반적으로 더 자주 사용하는 방법 중 하나입니다. 상위 구성 요소 A는 props 매개 변수를 통해 하위 구성 요소 B에 데이터를 전달하고, 구성 요소 B는 $emit를 통해 구성 요소 A에 이벤트를 보냅니다. (매개변수 데이터 전달) 구성요소 A는 B가 A로 전송한 데이터를 얻기 위해 $emit에 의해 트리거된 이벤트를 수신합니다. 구현 단계를 자세히 설명하겠습니다.

1: 상위 구성 요소가 하위 구성 요소에 값을 전달합니다.

// App.vue 父组件
<template>
    <a-compontent></a-compontent>
</template>
<script>
import aCompontent from &#39;./components/A.vue&#39;;
export default {
    name: &#39;app&#39;,
    compontent: { aCompontent },    
    data () {        
        return {
         dataA: &#39;dataA数据&#39;
       }
    }
}
// aCompontent 子组件
<template>
    <p>{{dataA}} // 在子组件中把父组件传递过来的值显示出来

<script>export default {
    name: &#39;aCompontent&#39;,
    props: {
        dataA: {           
              //这个就是父组件中子标签自定义名字            
              type: String,
            required: true  // 或者false
        }
    }
}
</script>
로그인 후 복사

2: 하위 구성 요소가 이벤트를 통해 상위 구성 요소에 값을 전달합니다.

// 子组件
<template>
    <p>点击向父组件传递数据</p>
</template>
<script>export default {
    name: &#39;child&#39;,
    methods:{        
        changeTitle() {
              // 自定义事件,会触发父组件的监听事件,并将数据以参数的形式传递
            this.$emit(&#39;sendDataToParent&#39;,&#39;这是子组件向父组件传递的数据&#39;); 
        }
    }
}

// 父组件
<template>
    <child @sendDataToParent="getChildData">

<script>
import child from &#39;./components/child.vue&#39;;
    export default {
    name: &#39;child&#39;,
    methods:{
        getChildData(data) {
            console.log(data); // 这里的得到了子组件的值
        }
    }
}
</script>
로그인 후 복사

2. $emit / $on

이 방법은 App.vue와 유사한 인스턴스를 모듈의 이벤트 센터로 사용하여 이벤트를 트리거하고 수신하는 데 사용합니다. App.vue에 넣으면 구성 요소에서 무엇이든 통신할 수 있지만 이 방법은 프로젝트 규모가 상대적으로 크면 유지 관리가 쉽지 않습니다.

예: 이제 Home.vue 및 A/B/C 구성 요소 4개가 있다고 가정합니다. 이 세 가지 구성 요소 AB는 형제 구성 요소입니다. Home.vue는 빈 Vue 인스턴스를 생성하고 통신 이벤트를 중단하는 상위 구성 요소와 동일합니다. . 이 인스턴스에 로드됨 -

D.js
import Vue from 'vue'export default new Vue()
로그인 후 복사
// 我们可以在router-view中监听change事件,也可以在mounted方法中监听
// home.vue<template>  <p>    <child-a></child-a>    <child-b></child-b>    <child-c></child-c>  </p></template>
로그인 후 복사
// A组件
<template>
  <p>将A组件的数据发送给C组件 - {{name}}</p>
</template>
<script>
import Event from "./D";export default {  data() {    return {
      name: &#39;Echo&#39;
    }
  },
  components: { Event },
  methods: {    dataA() {
      Event.$emit(&#39;data-a&#39;, this.name);
    }
  }
}
</script>
로그인 후 복사
// B组件
<template>
  <p>将B组件的数据发送给C组件 - {{age}}</p>
</template>
<script>
import Event from "./D";export default {  data() {    return {
      age: &#39;18&#39;
    }
  },
  components: { Event },
  methods: {    dataB() {
      Event.$emit(&#39;data-b&#39;, this.age);
    }
  }
}
</script>
로그인 후 복사
// C组件
<template>
  <p>C组件得到的数据 {{name}} {{age}}</p>
</template>
<script>
import Event from "./D";export default {  data() {    return {
      name: &#39;&#39;,
      age: &#39;&#39;
    }
  },
  components: { Event },  mounted() {
    // 在模板编译完成后执行
    Event.$on(&#39;data-a&#39;, name => {
      this.name = name;
    })
    Event.$on(&#39;data-b&#39;, age => {
      this.age = age;
    })
  }
}
</script>
로그인 후 복사

위에서 알 수 있듯이 C 구성요소의 마운트된 이벤트에서 A/B의 $emit 이벤트가 모니터링되고 이에 의해 전달된 매개변수를 가져옵니다(언제 실행될지 확실하지 않기 때문). 이벤트가 발생하므로 일반적으로 마운트/생성시 모니터링됩니다)

3. Vuex

Vuex는 상태 관리 모델입니다. 중앙 집중식 저장소를 사용하여 애플리케이션의 모든 구성 요소 상태를 관리하고 해당 규칙을 사용하여 상태가 예측 가능한 방식으로 변경되도록 합니다. Vuex 애플리케이션의 핵심은 스토어(창고, 컨테이너)입니다. 스토어에는 애플리케이션의 대부분의 상태가 포함되어 있습니다.

이 부분은 자세히 소개되지 않으며 공식 문서는 매우 상세합니다. vuex.vuejs.org/ zh /guide/st…

IV.$attrs / $listeners

위 그림과 같이 A/C 구성 요소가 어떻게 통신하는지 살펴보겠습니다. 이제 다음과 같은 해결책을 생각해 볼 수 있습니다.

  1. 데이터 관리를 위해 Vuex를 사용하지만, vuex 사용의 문제점은 프로젝트가 상대적으로 작고 구성 요소 간 공유 상태가 덜한 경우 vuex를 사용하는 것은 닭을 죽이는 것과 같다는 것입니다. 칼.
  2. B 구성 요소를 전송 스테이션으로 사용합니다. A 구성 요소가 C 구성 요소에 정보를 전달해야 할 경우 B는 A 구성 요소의 정보를 받은 다음 props를 사용하여 C 구성 요소에 전달합니다. 코드 유지 관리가 어렵습니다. C의 상태 변경을 A에 전달해야 하는 경우 이벤트 시스템을 사용하여 레벨별로 전달해야 합니다.

Vue2.4에서는 이러한 요구 사항을 해결하기 위해 attrs와 Listener가 도입되었고, InheritAttrs 옵션이 추가되었습니다. (아래 그림 참조)

$attrs의 역할은 경우에 따라 상속 Attrs와 함께 사용해야 합니다.

4가지 구성 요소가 있습니다: App.vue / child1.vue / child2.vue / child3 .vue, 이 4개의 구성 요소는 순서대로 중첩됩니다.

// App.vue
<template>
  <p>
    </p>
<p>App.vue</p>
<hr>
    // 这里我们可以看到,app.vue向下一集的child1组件传递了5个参数,分别是name / age / job / sayHi / title
    <child1></child1>
  
</template>
<script>
const child1 = () => import("./components/child1.vue");
export default {
  name: &#39;app&#39;,
  components: { child1 },  data() {    return {
      name: "Echo",
      age: "18",
      job: "FE",
      say: "this is Hi~"
    };
  }
};
</script>
로그인 후 복사
// child1.vue
<template>
  <p>
    </p>
<p>child1.vue</p>
    <p>name: {{ name }}</p>
    <p>childCom1的$attrs: {{ $attrs }}</p>
    <p>可以看到,$attrs这个对象集合中的值 = 所有传值过来的参数 - props中显示定义的参数</p>
    <hr>
    <child2></child2>
  
</template>
<script>
const child2 = () => import("./child2.vue");
export default {
  components: {
    child2
  },
  // 这个inheritAttrs默认值为true,不定义这个参数值就是true,可手动设置为false
  // inheritAttrs的意义在用,可以在从父组件获得参数的子组件根节点上,将所有的$attrs以dom属性的方式显示
  inheritAttrs: true, // 可以关闭自动挂载到组件根元素上的没有在props声明的属性
  props: {
    name: String // name作为props属性绑定
  },  created() {
    // 这里的$attrs就是所有从父组件传递过来的所有参数 然后 除去props中显式定义的参数后剩下的所有参数!!!
    console.log(this.$attrs); //  输出{age: "18", job: "FE", say-Hi: "this is Hi~", title: "App.vue的title"}
  }
};
</script>
로그인 후 복사
// child2.vue
<template>
  <p>
    </p>
<p>child2.vue</p>
    <p>age: {{ age }}</p>
    <p>childCom2: {{ $attrs }}</p>
    <hr>
    <child3></child3>
  
</template>
<script>
const child3 = () => import("./child3.vue");
export default {
  components: {
    child3
  },
  // 将inheritAttrs设置为false之后,将关闭自动挂载到组件根元素上的没有在props声明的属性
  inheritAttrs: false,
  props: {
    age: String
  },  created() {
    // 同理和上面一样,$attrs这个对象集合中的值 = 所有传值过来的参数 - props中显示定义的参数
    console.log(this.$attrs);
  }
};
</script>
로그인 후 복사
// child3.vue
<template>
  <p>
    </p>
<p>child3.vue</p>
    <p>job: {{job}}</p>
    <p>title: {{title}}</p>
    <p>childCom3: {{ $attrs }}</p>
  
</template>
<script>export default {
  inheritAttrs: true,
  props: {
    job: String,
    title: String
  }
};
</script>
로그인 후 복사

구체적인 디스플레이 효과를 살펴보겠습니다.

그리고 $listeners를 사용하는 방법에 대해 공식 문서에는 다음과 같이 나와 있습니다. 상위 범위(.native 수정자 없음) 장치에서 청취하는 v-on 이벤트가 포함되어 있습니다. v-on="$listeners"를 통해 내부 구성요소로 전달될 수 있습니다. 더 높은 수준의 구성요소를 생성할 때 매우 유용합니다! 문자 그대로의 의미에서 값을 수락해야 하는 상위 구성 요소에 청취 이벤트를 추가해야 합니까? 말할 것도 없이, 코드

는 여전히 순서대로 중첩된 3개의 구성요소입니다

<template>
  <p>
    <child2></child2>
  </p>
</template>
<script>
const child2 = () => import("./child2.vue");
export default {
  components: {
    child2
  },
  methods: {    reciveRocket() {
      console.log("reciveRocket success");
    }
  }
};
</script>复制代码
로그인 후 복사
<template>
  <p>
    <child3></child3>
  </p>
</template>
<script>
const child3 = () => import("./child3.vue");
export default {
  components: {
    child3
  },  created() {
    this.$emit(&#39;child2&#39;, &#39;child2-data&#39;);
  }
};
</script>复制代码
로그인 후 복사
// child3.vue
<template>
  <p>
    </p>
<p>child3</p>
  
</template>
<script>
export default {
  methods: {    startUpRocket() {
      this.$emit("upRocket");
      console.log("startUpRocket");
    }
  }
};
</script>复制代码
로그인 후 복사

这里的结果是,当我们点击 child3 组件的 child3 文字,触发 startUpRocket 事件,child1 组件就可以接收到,并触发 reciveRocket 打印结果如下:

> reciveRocket success
> startUpRocket
로그인 후 복사

五. $parent / $children 与 ref

  • ref:如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子组件上,引用就指向组件实例
  • $parent / $children:访问父 / 子实例

这两种方式都是直接得到组件实例,使用后可以直接调用组件的方法或访问数据。

我们先来看个用 ref 来访问组件的:

// child1子组件
export default {  
        data() {    
            return {
      title: 'Vue.js'
    };
  },
  methods: {    
      sayHello() {
      console.log('child1!!');
    }
  }
};
로그인 후 복사
// 父组件
<template>
  <child1></child1>
</template>
<script>  
export default {
    methods: {      
        sayHi () {
        const child1 = this.$refs.child1;
        console.log(child1.title);  // Vue.js
        child1.sayHello();  // 弹窗
      }
    }
  }
</script>
로그인 후 복사

六. provide/inject

provide/inject 是 Vue2.2.0 新增 API,这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。如果你熟悉 React,这与 React 的上下文特性很相似。

provide 和 inject 主要为高阶插件/组件库提供用例。并不推荐直接用于应用程序代码中。

由于自己对这部分的内容理解不是很深刻,所以感兴趣的可以前往官方文档查看: cn.vuejs.org/v2/api/#pro…

总结

常见使用场景可以分为三类:

  1. 父子通信:props / $emit;$parent / $children;$attrs/$listeners;provide / inject API; ref
  2. 兄弟通信:Vuex
  3. 跨级通信:Vuex;$attrs/$listeners;provide / inject API
    4.接下来我还会在我的裙里用视频讲解的方式给大家讲解【以下图中的内容有兴趣的可以来我的扣扣裙 519293536 免费交流学习,我都会尽力帮大家哦

推荐教程:《JS教程

위 내용은 Vue 구성요소와 통신하는 6가지 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
vue
원천:cnblogs.com
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿