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

组件
<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)
}
使用时,控制台抛出警告

由于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
在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的语法糖,看起来会让数据流向显得更加明确,恰好弥补这个缺点。
Vue - 自定义组件双向绑定的更多相关文章
- vue 父子组件双向绑定
vue组件有2大特性: 1.全局组件和局部组件 2.父子组件的数据传递 接下来直接用demo直接看如何传值(静态传值) father.vue <template> <div> ...
- 组件的通信 :provide / inject 对象进入后,就等于不用props,然后内部对象,直接复制可以接受数组,属性不能直接复制,可以用Object.assgin覆盖对象,或者Vue的set 双向绑定数据
组件的通信 :provide / inject 对象进入后,就等于不用props,然后内部对象,直接复制可以接受数组,属性不能直接复制,可以用Object.assgin覆盖对象,或者Vue的set 双 ...
- Vue自定义组件实现v-model指令
Tips: 本文所描述的Vue均默认是Vue2版本 在我们初次接触Vue的时候,一定会了解到一个语法糖,那就是v-model指令,它带给我们的第一印象就是它可以实现双向绑定 那么,什么是双向绑定?通俗 ...
- Vue的数据双向绑定和Object.defineProperty()
Vue是前端三大框架之一,也被很多人指责抄袭,说他的两个核心功能,一个数据双向绑定,一个组件化分别抄袭angular的数据双向绑定和react的组件化思想,咱们今天就不谈这种大是大非,当然我也没到达那 ...
- Vue框架之双向绑定事件
Vue框架之双向绑定事件 首先介绍下Vue框架的语法 vue通过 {{temp}} 来渲染变量 {{count+100}} # 求和 v-text # 为标签插入text文本 v-html # 为标签 ...
- Angular中父子组件双向绑定传值
下面为大家展示一个较为简单的ng父子组件双向绑定传值,下面是父组件页面 这个页面的大概功能就是父组件(红色)通过输入框输入内容反映到子组件上进行展示,并且进行了投影, 子组件(橙黄色)通过Input输 ...
- 用ES6的class模仿Vue写一个双向绑定
原文地址:用ES6的class模仿Vue写一个双向绑定 点击在线尝试一下 最终效果如下: 构造器(constructor) 构造一个TinyVue对象,包含基本的el,data,methods cla ...
- vue自定义组件(vue.use(),install)+全局组件+局部组件
相信大家都用过element-ui.mintui.iview等诸如此类的组件库,具体用法请参考:https://www.cnblogs.com/wangtong111/p/11522520.html ...
- 原生js实现 vue的数据双向绑定
原生js实现一个简单的vue的数据双向绑定 vue是采用数据劫持结合发布者-订阅者模式的方式,通过Object.defineProperty()来劫持各个属性的setter,getter,在数据变动时 ...
随机推荐
- ng-核心特性(模型概念)
angular核心特性 很多开发者已经做过非常多的项目,但是当你跟他聊的时候,你很快就会发现他并没有掌握这门框架的精髓.打几个比方,当别人提到 Spring 的时候,你的大脑里面第一个想到一定是 DI ...
- 使用FDATOOL生成xilinx中FIR滤波器IP核的系数
1.在MATLAB命令窗口输入fdatool后回车,打开“Filter Designer & Analysis Tool”工具界面: 2.点击左下角的Set quantization para ...
- npm命令笔记-----转自网络,仅供自己查看使用
npm是一个node包管理和分发工具,已经成为了非官方的发布node模块(包)的标准.有了npm,可以很快的找到特定服务要使用的包,进行下载.安装以及管理已经安装 的包. npm常用指令 1.npm ...
- pytest-测试用例teardown和setup
setup和teardown对于处理一些前置条件很有帮助 用例运行级别 模块级(setup_moudle/teardown_moudle)开始于模块始末,全局的 所有用例开始前/结束后执行一次(整个. ...
- nginx中部署前端,后端打成jar包运行
项目是前后端分离:前端用vue开发,后端用的是springboot开发 会产生跨域问题,故在前端里用了代理 1.本前端项目是用vue开发: 1.1打包:终端 vscode快捷键:crtl+~ 然后n ...
- 使用yum安装报错:[Errno 256] No more mirrors to try
背景:我使用yum方式安装软件时,比如zabbix这种软件,我们在安装时一般都是直接到zabbix官网,按照官方的步骤进行安装,但是有一个问题,官方的服务器不在国内,时常会在安装时导致超时报错.此时解 ...
- [CF891C] Envy - Kruskal,并查集
给出一个 n 个点 m条边的无向图,每条边有边权,共 Q次询问,每次给出 \(k\)条边,问这些边能否同时在一棵最小生成树上. Solution 所有最小生成树中某权值的边的数量是一定的 加完小于某权 ...
- dataTables插件的使用
用到dataTables这个插件还是因为Metronic这个框架里有用到,不然我不会选择它的,为啥呢?就感觉它的文档有点复杂(当然,也有我智商不够用的原因):既然用了,那就说说我遇到的问题吧,以防下次 ...
- SpringMVC流程图示
- 让Surface Shader不受光照的明暗影响
直接上码 Shader "Custom/3DVideo" { Properties { _Color (,,,) _MainTex ("Albedo (RGB)" ...