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 相关文章推荐
FireFox与IE 下js兼容触发click事件的代码
Nov 20 Javascript
javascript深入理解js闭包
Jul 03 Javascript
超简单JS二级、多级联动的简单实例
Feb 18 Javascript
jQuery制作拼图小游戏
Jan 12 Javascript
js实现进度条的方法
Feb 13 Javascript
利用JavaScript阻止表单提交的两种方法
Aug 11 Javascript
vue+vuecli+webpack中使用mockjs模拟后端数据的示例
Oct 24 Javascript
微信小程序实现提交input信息到后台的方法示例
Jan 19 Javascript
文章或博客自动生成章节目录索引(支持三级)的实现代码
May 10 Javascript
浅谈vue中使用编辑器vue-quill-editor踩过的坑
Aug 03 Javascript
JavaScript点击按钮生成4位随机验证码
Jan 28 Javascript
vue使用echarts实现折线图
Mar 21 Vue.js
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
PHP实现用户认证及管理完全源码
2007/03/11 PHP
使用PHP会话(Session)实现用户登陆功能
2013/06/29 PHP
PHP生成随机密码类分享
2014/06/25 PHP
php基于GD库画五星红旗的方法
2015/02/24 PHP
php版微信发红包接口用法示例
2016/09/23 PHP
JavaScript 类型的包装对象(Typed Wrappers)
2011/10/27 Javascript
Jquery Ajax xmlhttp请求成功问题
2015/02/04 Javascript
使用Browserify配合jQuery进行编程的超级指南
2015/07/28 Javascript
微信小程序 开发之快递查询功能的实现
2017/01/09 Javascript
jQuery实现简单的手风琴效果
2020/04/17 jQuery
基于Vue组件化的日期联动选择器功能的实现代码
2018/11/30 Javascript
vue移动端下拉刷新和上滑加载
2020/10/27 Javascript
JS实现点击掉落特效
2021/01/29 Javascript
微信小程序组件生命周期的踩坑记录
2021/03/03 Javascript
[03:58]2014DOTA2国际邀请赛 龙宝赛后解密DK获胜之道
2014/07/14 DOTA
[01:11:27]2018DOTA2亚洲邀请赛小组赛 A组加赛 Newbee vs Optic
2018/04/03 DOTA
在Python的Flask框架中构建Web表单的教程
2016/06/04 Python
django模板结构优化的方法
2019/02/28 Python
详解Python字典的操作
2019/03/04 Python
详解python:time模块用法
2019/03/25 Python
Python利用sqlacodegen自动生成ORM实体类示例
2019/06/04 Python
Pandas中resample方法详解
2019/07/02 Python
Python3 pandas 操作列表实例详解
2019/09/23 Python
解决Python二维数组赋值问题
2019/11/28 Python
QT5 Designer 打不开的问题及解决方法
2020/08/20 Python
美国户外生活方式品牌:Eddie Bauer
2016/12/28 全球购物
Sunglasses Shop丹麦:欧洲第一的太阳镜在线销售网站
2017/10/22 全球购物
Under Armour安德玛法国官网:美国高端运动科技品牌
2018/06/29 全球购物
项目专员岗位职责
2013/12/04 职场文书
初中同学聚会感言
2014/02/11 职场文书
电子工程专业毕业生求职信
2014/03/14 职场文书
揭牌仪式策划方案
2014/05/28 职场文书
员工工作自我评价
2014/09/26 职场文书
OpenCV3.3+Python3.6实现图片高斯模糊
2021/05/18 Python
解决Python中的modf()函数取小数部分不准确问题
2021/05/28 Python
《原神》新角色演示“神里绫人:林隐泓洄” 宠妹狂魔
2022/04/03 其他游戏