不管在任何的语言或框架中,咱们都提倡代码的复用性。对于Vue来讲也是如此,相同的代码逻辑会被封装成组件,除了复用以外,更重要的是统一管理提升开发效率。我真就接手过一个项目,多个页面都会用到的列表,没有去封装列表组件,只要有一点改动,每一个页面都得加上。很确定的说,没有用组件化开发的Vue项目是没有灵魂的。因此如何封装一个优雅且复用性高的组件成为咱们必需的技能。html
首先来看一个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,而是由组件内部去更新。api
可能有人会想到,既然要由内部管理,那在组件内部修改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%81ide
组件model选项组件化
容许一个自定义组件在使用 v-model 时定制 prop 和 event。默认状况下,一个组件上的 v-model 会把 value 用做 prop 且把 input 用做 event,可是一些输入类型好比单选框和复选框按钮可能想使用 value prop 来达到不一样的目的。使用 model 选项能够回避这些状况产生的冲突。ui
model: https://cn.vuejs.org/v2/api/#modelthis
在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。code
使用组件双向绑定后,属性在组件内部被更新时,父组件的 activeName 也会随之更新,这样使用者能够很明确的知道数据可能会被修改。
<Tabs :tabs="tabs" v-model="activeName" />
使用组件的model选项实现自定义组件双向绑定,在组件内部经过事件更新属性值,这样的自定义组件使用起来更优雅。其实经过model选项的方式去修改父级属性,我认为有点违反了单向数据流的原则。原本单向数据流是不容许子级修改父级属性的,只是使用v-model的语法糖,看起来会让数据流向显得更加明确,刚好弥补这个缺点。