Vue中的.sync 修饰符

209 阅读1分钟

.sync 修饰符可以让子组件改变父组件状态的代码更容易被区分,但是它只是作为一个编译时的语法糖存在。它会被扩展为一个自动更新父组件属性的 v-on 监听器。它可以更方便的父子组件间的双向绑定。

在解释.sync 修饰符之前,我们需要了解 vue 之父子组件间是怎么样通信的。

通过 prop 实现通信

子组件的 props 选项能够接收来自父组件数据。没错,仅仅只能接收,props 是单向绑定的,即只能父组件向子组件传递,不能反向。

子组件通过 props 选项来声明一个自定义的属性,然后父组件就可以在嵌套标签的时候,通过这个属性往子组件传递数据了。但是我们更多的情况需要动态的数据。这时候就可以用 v-bind 来实现。通过 v-bind 绑定 props 的自定义的属性,传递去过的就不是静态的字符串了,它可以是一个表达式、布尔值、对象等等任何类型的值。

<!-- 父组件 -->
<template>
  <div>
    <h1>我是父组件!</h1>
    <child message="我是子组件一!"></child>

    <!-- 这是一个 JavaScript 表达式而不是一个字符串。-->
    <child v-bind:message="a+b"></child>

    <!-- 用一个变量进行动态赋值。-->
    <child v-bind:message="msg"></child>
  </div>
</template>

<script>
  import Child from "../components/child.vue";
  export default {
    components: { Child },
    data() {
      return {
        a: "我是子组件二!",
        b: 112233,
        msg: "我是子组件三!" + Math.random()
      };
    }
  };
</script>
<!-- 子组件 -->
<template>
  <h3>{{message}}</h3>
</template>
<script>
  export default {
    props: ["message"]
  };
</script>

出来的效果是这样的:

通过$emit 实现通信

props 主要是父组件向子组件通信,而通过$emit 实现子组件向父组件通信。

$emit 绑定一个自定义事件 event,当这个这个语句被执行到的时候,就会将参数 arg 传递给父组件,父组件通过@event 监听并接收参数。

<template>
  <div>
    <h1>{{title}}</h1>
    <child @getMessage="showMsg"></child>
  </div>
</template>

<script>
  import Child from "../components/child.vue";
  export default {
    components: { Child },
    data() {
      return {
        title: ""
      };
    },
    methods: {
      showMsg(title) {
        this.title = title;
      }
    }
  };
</script>
<template>
  <h3>我是子组件!</h3>
</template>
<script>
  export default {
    mounted: function() {
      this.$emit("getMessage", "我是父组件!");
    }
  };
</script>

.sync 修饰符

通过将 props 和$emit 结合起来就可以实现父子组件的双向通信。

<div id="app">
  <div>{{bar}}</div>
  <my-comp :foo.sync="bar"></my-comp>
  <!-- <my-comp :foo="bar" @update:foo="val => bar = val"></my-comp> -->
</div>

<script>
  new Vue({
    el: "#app",
    data: { bar: 0 }
  });

  Vue.component("my-comp", {
    template: '<h3>{{foo}}</h3><div @click="increment">点我+1</div>',
    data: function() {
      return { copyFoo: this.foo };
    },
    props: ["foo"],
    methods: {
      increment: function() {
        this.$emit("update:foo", ++this.copyFoo);
      }
    }
  });
</script>

子组件通过“update:foo”事件将讲一个值传给父组件,父组件接收后再将这个值赋给变量“foo”,接着在通过 props 将这个“foo”,子组件又接收到了新值。就实现了双向通信。

在使用了.sync 后,

<comp :foo="bar" @update:foo="val => bar = val"></comp>

被简化为

<my-comp :foo.sync="bar"></my-comp>

可以看出 sync 就是一个语法糖,简化了父子组件双向通信的写法。