• Vue中v-model解析、sync修饰符解析


    上善若水,水善利萬物而不爭。——《道德經》
    简介
    
    在平时开发是经常用到一些父子组件通信,经常用到props、vuex等等,这里面记录另外的三种方式v-model、sync是怎么使用,再说是怎么实现,其实v-model、sync都是语法糖。还有$attr、$listener实现父子组件通信。
    使用方式
    v-model
    
        2.2.0+ 新增
    
    v-mode1其实就是一个语法糖,默认会利用名为value的props和名为input的事件,但是像单选框、复选框等类型的输入龙剑可能会讲value特性用于不同的目的。
    
    v-model的使用场景:当子组件需要改变父组件通过props传入的值
    
    父组件
    
        父组件通过v-model绑定值
        如需根据v-model传入的值改变,而触发其他更新请通过watch传入的值
    
    子组件
    
        声明model对象 设置事件event和prop字段
        通过porps接受父组件传送值
        修改是通过this.$emit广播事件
    
    代码示例:
    
    父组件代码
    
    <template>
      <children v-model="message"></children>
    </template>
    <script>
    import children from "./children.vue";
    export default {
      components: {
        children
      },
      data() {
        return {
          message: "parent"
        };
      },
      watch: {
        // 监听message变化
        message(newV, oldV) {
          console.log(newV, oldV);
        }
      }
    };
    </script>
    
    子组件代码
    
    <template>
      <h1>{{ message }}</h1>
    </template>
    <script>
    export default {
      model: {
        prop: "message", //这个字段,是指父组件设置 v-model 时,将变量值传给子组件的 msg
        event: "input" //这个字段,是指父组件监听 parent-event 事件
      },
      props: {
        message: String //此处必须定义和model的prop相同的props,因为v-model会传值给子组件
      },
      mounted() {
        //这里模拟异步将msg传到父组件v-model,实现双向控制
        setTimeout(_ => {
          this.$emit("input", "children");
          //将这个值通过 emit 触发parent-event,将some传递给父组件的v-model绑定的变量
        }, 1500);
      }
    };
    </script>
    
    上面这个示例是通过v-model实现的,下面不通过v-model实现同样效果。
    不使用 v-model 实现
    
    代码示例如下:
    
    父组件代码修改
    
    <template>
      <Children :message="message" @input="(event) => { message = event }"/>
    </template>
    <script>
    // 不变
    </script>
    
    子组件代码修改
    
    <template>
      // 不变
    </template>
    <script>
    export default {
      props: {
        message: String
      },
      mounted() {
        setTimeout(() => {
          this.$emit("input", "children");
        }, 1500);
      }
    };
    </script>
    
    只是把v-model拆分为props和@input事件,子组件不需要配置model,只需要接受props和通过this.$emit广播事件就可以。 当然这个相对于v-model方法比较简便,但是灵活度查很多,选择使用那种看个人喜好。 在线地址:
    
    不能放iframe只能放一放一个链接了本篇代码实例
    sync
    
        2.3.0+ 新增
    
    在有些情况下,我们可能需要对一个 prop 进行**“双向绑定”。不幸的是,真正的双向绑定**会带来维护上的问题,因为子组件可以修改父组件,且在父组件和子组件都没有明显的改动来源。
    
    这也是为什么我们推荐以 update:myPropName 的模式触发事件取而代之。同时也可以通过sync修饰符来实现。
    
    在上面代码的基础上大致修改如下:
    
    父组件
    
        通过修改触发事件input为update:myPropName实现相同效果
    
    子组件
    
        通过修改this.$emit(update:myPropName)
    
    代码如下:
    
    父组件代码修改
    
      // 修改如下
      <Children :message="message" @update:input="(event) => { message = event }"/>
    
    子组件代码修改
    
      // 其他不变
      this.$emit("update:input", "children");
    
    sync实现
    
    上面的代码可以通过sync简写为下面代码:
    
    父组件代码修改
    
      // 修改如下
      <Children :messag.sync="message"/>
    
    子组件代码修改
    
      // 其他不变
      this.$emit("update:messag", "children");
    
    同时sync也支持对象,要配合v-bind实现可以简写为:,但是要注意这个对象如下两条:
    
        注意带有 .sync 修饰符的 v-bind 不能和表达式一起使用 (例如 v-bind:title.sync=”doc.title + ‘!’” 是无效的)。取而代之的是,你只能提供你想要绑定的属性名,类似 v-model。 将 v-bind.sync 用在一个字面量的对象上,例如 v-bind.sync=”{ title: doc.title }”,是无法正常工作的,因为在解析一个像这样的复杂表达式的时候,有很多边缘情况需要考虑。
    
    attrs、listeners
    $attrs
    
        2.4.0 新增
    
        类型:{ [key: string]: string }
        只读
        详细: 包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定 (class 和 style 除外),并且可以通过 v-bind="$attrs" 传入内部组件——在创建高级别的组件时非常有用。
    
    $listeners
    
        2.4.0 新增
    
        类型:{ [key: string]: Function | Array<Function> }
        只读
        详细: 包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件——在创建更高层次的组件时非常有用。
    
    实现通信
    
    实现父子组件通信
    
    父组件代码
    
    <template>
      <div class="parent">
        <Children
          :message="message"
          @upDate="upDate"
          type="del"
          @input="(event) => { message = event }"
        />
      </div>
    </template>
    
    <script>
    import Children from "./Children";
    export default {
      components: {
        Children
      },
      data() {
        return {
          message: "parent",
          type: "del"
        };
      },
      methods: {
        upDate (event) {
          console.log(event);
          this.type = event;
        }
      },
      watch: {
        message: function() {
          console.log("更新message值为" + this.message);
        }
      }
    };
    </script>
    
    子组件代码
    
    <template>
      <div v-bind="$attrs" v-on="$listeners" class="children">{{message}} <span @click="$listeners.upDate('data')">{{$attrs.type}}</span></div>
    </template>
    
    <script>
    export default {
      props: {
        message: String
      },
      mounted() {
        // console.log(this.$attrs);
        // console.log(this.$listeners);
        setTimeout(() => {
          this.$emit("input", "children");
          this.$emit('upDate', 'add')
        }, 1500);
      }
    };
    </script>
    
    同时$attrs、$listeners都是可以跨域父子组件,可以父子子子组件传递,类似于react中的context,只是一部分设计理念相同。
    总结
    
    其实就是检测到.sync修饰符,在complier阶段会编译生成多个prop,生成多个事件。其实像这个指令、修饰符、自定义指令都是在vue编译是解析成为v8能执行的代码。
    
    无论是vue、babel、react的complier编译阶段大致分为三个阶段:
    
        通过词法解析parse生成抽象AST或抽象代码树
        优化AST,比如vue标记静态节点,babal中抽取静态代码,这个阶段被称为optimize或者优化AST树
        在AST代码的阶段上,生成可执行代码,这个过程可以叫做codegen
    
    v-model、sync都可以实现父子组件通信,并且可以在子组件中修改父组件传入的值。在平常看法的时候进场可以用到这两种方式,具体选择那种方式看个人喜好。在element-ui这个input组件也用到相关的属性。
  • 相关阅读:
    C#中添加文本框的上标及文字大小
    综采工作面设备接替计划管理系统
    一位年轻女董事长的27条忠告
    开发人员一定要加入收藏夹的网站
    哈佛图书馆墙上的名训
    加密、解密.NET字符串
    C#文件上传下载
    DotNetNuke3.0.8 简体中文语言包(可直接导入版)发布
    DNN的Core Team
    DotNetNuke3.0.8文件管理器的错误及解决方法
  • 原文地址:https://www.cnblogs.com/Free-Thinker/p/11676622.html
Copyright © 2020-2023  润新知