Vue自定义组件双向绑定实现原理及方法详解


Posted in Javascript onSeptember 03, 2020

前言

无论在任何的语言或框架中,我们都提倡代码的复用性。对于Vue来说也是如此,相同的代码逻辑会被封装成组件,除了复用之外,更重要的是统一管理提高开发效率。我真就接手过一个项目,多个页面都会用到的列表,没有去封装列表组件,只要有一点改动,每个页面都得加上。很肯定的说,没有用组件化开发的Vue项目是没有灵魂的。所以如何封装一个优雅且复用性高的组件成为我们必需的技能。

Tab自定义组件

首先来看一个Tab组件的实现,看看它存在什么问题,哪里可以改进?

效果

Vue自定义组件双向绑定实现原理及方法详解

组件

<template>
 <div class="tabs">
  <div 
   class="tab-item" 
   :class="{'tab--active':item===activeName}"
   v-for="(item,index) in tabs" 
   :key="index" 
   @click="tabChange(item)">
   {{item}}
  </div>
 </div>
</template>

<script>
export default {
 props:{
  tabs:{
   type: Array,
   default: ()=> []
  },
  activeName:{
   type: String,
   default: ''
  }
 },
 methods:{
  tabChange(item){
   this.$emit('tabChange',item)
  }
 },
}
</script>

使用

<template>
 <div>
  <Tabs :tabs="tabs" :activeName="activeName" @tabChange="tabChange" />
 </div>
</template>

<script>
import Tabs from '../components/Tabs'
export default {
 components:{
  Tabs
 },
 data(){
  return{
   tabs:['黄金体验','败者食尘','绯红之王','白金之星','波纹疾走'],
   activeName: '黄金体验'
  }
 },
 methods:{
  tabChange(item){
   this.activeName = item
  }
 },
}
</script>

这个组件最大的问题就是,activeName 需要使用者额外通过事件来手动更新,假如有另一个使用者接手,在不知道这种情况下使用,会出现tab没有切换的情况。然后要去看组件内部实现,再回来修改代码,很显然这样的组件是失败的。本着所有的脏活累活都由组件实现的原则,理想的状态应该是使用者不需要管理 activeName,而是由组件内部去更新。

如何改进修改prop?

可能有人会想到,既然要由内部管理,那在组件内部修改prop的值是不是就可以了?来看下这样的做法是否可行
修改组件tabChange方法,在点击时更新prop的值

tabChange(item){
 this.activeName = item
 this.$emit('tabChange',item)
}

使用时,控制台抛出警告

Vue自定义组件双向绑定实现原理及方法详解

由于prop是单向数据流,父级prop的更新会向下流动到子组件中,相反的在子组件内部直接更新状态,会导致数据的流向不明确。例如,在父组件中有多个子组件依赖同一个属性,其中一个子组件更新该属性,会引发其余子组件发生改变,发生问题时不容易被找到,因此Vue不推荐我们这样做。另外,在父组件发生更新时,子组件的prop会被刷新为最新的值。

单向数据流: https://cn.vuejs.org/v2/guide/components-props.html#%E5%8D%95%E5%90%91%E6%95%B0%E6%8D%AE%E6%B5%81

正解:model选项改进组件

组件model选项

允许一个自定义组件在使用 v-model 时定制 prop 和 event。默认情况下,一个组件上的 v-model 会把 value 用作 prop 且把 input 用作 event,但是一些输入类型比如单选框和复选框按钮可能想使用 value prop 来达到不同的目的。使用 model 选项可以回避这些情况产生的冲突。

model: https://cn.vuejs.org/v2/api/

在model选项里,我们可以绑定一个属性,并为其添加事件,只需在调用方法时传入值即可更新属性。

<script>
export default {
 model:{
  prop: 'activeName',
  event: 'update'
 },
 props:{
  tabs:{
   type: Array,
   default: ()=> []
  },
  activeName:{
   type: String,
   default: ''
  }
 },
 methods:{
  tabChange(item){
   this.$emit('update',item) // 这里更新activeName
   this.$emit('tabChange',item)
  }
 }
}
</script>

注意你仍然需要在组件的 props 选项里声明 prop。

使用

使用组件双向绑定后,属性在组件内部被更新时,父组件的 activeName 也会随之更新,这样使用者可以很明确的知道数据可能会被修改。

<Tabs :tabs="tabs" v-model="activeName" />

总结

使用组件的model选项实现自定义组件双向绑定,在组件内部通过事件更新属性值,这样的自定义组件使用起来更优雅。其实通过model选项的方式去修改父级属性,我认为有点违反了单向数据流的原则。本来单向数据流是不允许子级修改父级属性的,只是使用v-model的语法糖,看起来会让数据流向显得更加明确,恰好弥补这个缺点。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
在iframe里的页面编写js,实现在父窗口上创建动画效果展开和收缩的div(不变动iframe父窗口代码)
Dec 20 Javascript
JavaScript初学者应注意的七个细节小结
Jan 30 Javascript
js保留小数点后几位的写法
Jan 03 Javascript
JS组件Bootstrap Table表格多行拖拽效果实现代码
Dec 08 Javascript
jQuery EasyUI中DataGird动态生成列的方法
Apr 05 Javascript
Es6 写的文件import 起来解决方案详解
Dec 13 Javascript
canvas实现粒子时钟效果
Feb 06 Javascript
JS简单实现点击按钮或文字显示遮罩层的方法
Apr 27 Javascript
JS设计模式之观察者模式实现实时改变页面中金额数的方法
Feb 05 Javascript
使用layui 渲染table数据表格的实例代码
Aug 19 Javascript
JS闭包经典实例详解
Dec 20 Javascript
JS实现公告上线滚动效果
Jan 10 Javascript
Vue js with语句原理及用法解析
Sep 03 #Javascript
Vue通过provide inject实现组件通信
Sep 03 #Javascript
Vue组件通信$attrs、$listeners实现原理解析
Sep 03 #Javascript
Vue父组件监听子组件生命周期
Sep 03 #Javascript
JavaScript 几种循环方式以及模块化的总结
Sep 03 #Javascript
Vuejs通过拖动改变元素宽度实现自适应
Sep 02 #Javascript
Vue Object.defineProperty及ProxyVue实现双向数据绑定
Sep 02 #Javascript
You might like
一个分页的论坛
2006/10/09 PHP
超级简单的php+mysql留言本源码
2009/11/11 PHP
codeigniter教程之上传视频并使用ffmpeg转flv示例
2014/02/13 PHP
Codeigniter实现多文件上传并创建多个缩略图
2014/06/12 PHP
PHP生成图片验证码、点击切换实例
2014/06/25 PHP
php跨服务器访问方法小结
2015/05/12 PHP
php脚本守护进程原理与实现方法详解
2017/07/20 PHP
PHP远程连接oracle数据库操作实现方法图文详解
2019/04/11 PHP
JQery 渐变图片导航效果代码 漂亮
2010/01/01 Javascript
jquery下实现overlay遮罩层代码
2010/08/25 Javascript
js中document.write使用过程中的一点疑问解答
2014/03/20 Javascript
JS实现的颜色实时渐变效果完整实例
2016/03/25 Javascript
jQuery插件Flexslider实现图片轮播、图文结合滑动切换效果
2020/04/16 Javascript
react.js 翻页插件实例代码
2017/01/19 Javascript
详解AngularJS中$filter过滤器使用(自定义过滤器)
2017/02/04 Javascript
JavaScript闭包_动力节点Java学院整理
2017/06/27 Javascript
解决Vue.js 2.0 有时双向绑定img src属性失败的问题
2018/03/14 Javascript
element vue Array数组和Map对象的添加与删除操作
2018/11/14 Javascript
jQuery点击页面其他部分隐藏下拉菜单功能
2018/11/27 jQuery
nodeJS进程管理器pm2的使用
2019/01/09 NodeJs
JavaScript定时器设置、使用与倒计时案例详解
2019/07/08 Javascript
vue 避免变量赋值后双向绑定的操作
2020/11/07 Javascript
Request的中断和ErrorHandler实例解析
2018/02/12 Python
python判断字符串或者集合是否为空的实例
2019/01/23 Python
numpy 返回函数的上三角矩阵实例
2019/11/25 Python
Elasticsearch py客户端库安装及使用方法解析
2020/09/14 Python
python opencv实现直线检测并测出倾斜角度(附源码+注释)
2020/12/31 Python
html table呈现个人简历以及单元格宽度失效的问题解决
2021/01/22 HTML / CSS
台湾网友喜爱的综合型网路购物商城:Yahoo! 奇摩购物中心
2018/03/10 全球购物
智能电子秤、手表和健康监测仪:Withings(之前为诺基亚健康)
2018/10/30 全球购物
助人为乐表扬信范文
2014/01/14 职场文书
感恩寄语大全
2014/04/11 职场文书
个人考核材料
2014/05/15 职场文书
学院党委班子四风问题自查报告及整改措施
2014/10/25 职场文书
终止合同协议书范本
2016/03/22 职场文书
python用tkinter开发的扫雷游戏
2021/06/01 Python