网易首页 > 网易号 > 正文 申请入驻

3 个简单的技巧让你的 vue.js 代码更优雅!

0
分享至

关注 逆锋起笔 ,回复“ 加群 ”

加入我们一起学习,天天进步

作者:红尘炼心

https://juejin.cn/post/7005751368937897991

前言

近来入坑了一个Vue项目,感觉掉进了祖传屎山中,可读性极差,更别说可维护性了。故借此专栏提几点关于Vue代码可读性的建议,觉得有用的点个赞,觉得建议不合理的发表评论批评一下,有更好的建议欢迎发表评论补充一下。

一、善用组件让代码更有条理性

千万不要把一个页面的实现代码都梭哈在一个.vue文件中,除非这个页面非常简单,不然这个.vue文件中的代码会又长又臭。

Vue提供组件的目的不仅仅是为了复用,也可以用来分割代码,甚至善用组件可以优化页面的渲染更新速度。这是因为Vue页面渲染更新时不会去更新页面中的组件,除非组件的props或者slot所引用的数据发生变化。

可以按以下步骤来将一个Vue页面分割成一个个组件让代码更有条理性

1.1、提取UI组件

如何定义UI组件呢?个人建议按有无处理服务端数据来区分UI组件和业务组件。例如加载弹窗、二次确认弹窗、消息提示框等等属于UI交互组件。

将UI组件提取出来后,可以把UI交互的代码和业务交互的代码剥离开来。切记不能UI组件中写业务代码,这样UI组件将无法复用。

举一个反例,在二次确认弹窗中添加二次确认后要处理的业务代码,导致UI组件将无法复用。我们可以模仿ElementUI中二次确认弹窗的调用来实现一个二次确认弹窗组件。

this. $confirm(message, title, options)
. then(res =>{})
.catch(err =>{})

这样业务代码可以写在then的回调函数中,组件的核心实现代码如下所示:

//confirm.vue

"show">
//...

"ok">

"cancel">

//index.js
import Vue from 'vue';
import options from './confirm.vue';
const Confirm = Vue.extend(options);
let confirm = undefined;
const ConfirmInit = (options = {}) => {
return new Promise((resolve, reject) => {
options.resolve = resolve;
options.reject = reject;
confirm = new Confirm({
el: document.createElement( 'div'),
data: options
})
document.body.appendChild(confirm. $el);
Vue.nextTick(() => {
if (confirm) confirm.show = true;
})
return confirm;
})
}
Vue.prototype. $confirm = ConfirmInit;

//main.js
import 'components/confirm/index.js';//全局注册二次确认弹窗confirm组件

1.2、按模块提取业务组件

一个页面可以分为多个区域,比如头部、底部、侧边栏、商品列表、成员列表等等,每个区域可以当作一个模块来提取业务组件。

1.3、按功能提取功能组件

按模块提取完业务组件,此时业务组件有可能还是很庞大的,故要按功能在进一步地提取功能组件。

功能有大有小,提取要注意把握几个原则:


  • 过于简单的功能不提取

    例如一个收藏的功能,只要请求一个接口就完成,类似这样的功能不要提取。要有一定复杂度的逻辑操作的功能才提取。



  • 功能要单一,一个功能组件只处理一项业务。

    例如一个文件阅读器组件,有一个需求,要求打开文件后自动收藏该文件,那么收藏逻辑代码要写在哪里呢?

    或许你想都没想就在组件中监听文件成功打开的方法中写下收藏逻辑代码,过一段时间后,需求改为要先添加到阅读记录中再点击收藏按钮收藏,去组件中修改代码时发现另一个页面也引用了这个组件,故在组件中要额外加个参数做业务场景区分,随着需求的变化导致业务场景的叠加,组件的代码中会添加各种判断逻辑,久而久之变得又长又臭,显然这种做法是不可去。

    正确的做法是在组件标签上自定义一个事件on-fileOpen-success,用handleFileOpenSuccess函数来监听这个事件。

    @on-fileOpen-success="handleFileOpenSuccess"

    在组件中监听文件成功打开的方法中执行this.$emit('on-fileOpen-success',data)触发这个事件,其中data可以把文件信息传递出去,在handleFileOpenSuccess函数去处理收藏或者添加历史记录再收藏等业务交互。这种做法使文件阅读器组件具有单一性。



  • 功能组件尽量少包含UI部分,UI部分用slot插槽传入,这样使组件更纯粹,更具有复用性。

    例如上传组件的上传图标,不可能随着UI设计稿的变动就往里面添加一个上传图标,此时可以利用slot插槽把上传图标传入。

    //upload.vue


    "icon">

    //index.vue

    #icon>
    //上传图标

二、利用v-bind使组件的属性更具有可读性

如果想要将一个对象的所有属性都作为prop传入组件componentA,可以使用不带参数的v-bind。例如,对于一个给定的对象params

params: {
id: 1,
name: 'vue'
}

优化前

"params.id" :name= "params.name">

优化后

"params">

三、利用attrs与attrs与attrs与listeners来封装第三方组件 1、$attrs

在封装第三方组件中,经常会遇到一个问题,如何通过封装的组件去使用第三方组件自身的属性和事件。

比如封装一个elementUi组件中的Input输入框组件myInput,当输入错误的内容在输入框下面显示错误的提示。

myInput组件代码如下所示:


"input">

{{errorTip}}

这样调用myInput组件,其中errorTip为输入框输入错误的提示。

"input" :errorTip= "errorTip">

如果要在myInput组件上添加一个disabled属性来禁用输入框,要如何实现呢?一般同学会这么做


"input"
:disabled= "disabled">

{{errorTip}}

过一段时间后又要在myInput组件上添加el-input组件其它的属性,el-input组件总共有27个多,那该怎么呢,难道一个个用prop传进去,这样做不仅可读性差而且繁琐,可以用$attrs一步到位,先来看一下attrs的定义。

$attrs: 包含了父作用域中不作为 prop 被识别 (且获取) 的 attribute 绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定 (class 和 style 除外),并且可以通过 v-bind="$attrs" 传入内部组件


"input"
v-bind= "$attrs">

{{errorTip}}

这还不够,还得把inheritAttrs选项设置为false,为什么呢,来看一下inheritAttrs选项的定义就明白了。

默认情况下父作用域的不被认作 props 的 attribute 绑定 (attribute bindings) 将会“回退”且作为普通的 HTML attribute 应用在子组件的根元素上。当撰写包裹一个目标元素或另一个组件的组件时,这可能不会总是符合预期行为。通过设置 inheritAttrs 为 false,这些默认行为将会被去掉。而通过 $attrs 可以让这些 attribute 生效,且可以通过 v-bind 显性的绑定到非根元素上。注意:这个选项不影响 class 和 style 绑定。

简单来说,把inheritAttrs设置为falsev-bind="$attrs"才生效。


"input"
v-bind= "$attrs">

{{errorTip}}

这样就可以很清楚的把el-input组件的属性和myinput组件的属性区分开来了,组件的props选项的可读性大大提高。

2、$listeners

那么如何实现在myIpput组件上使用el-input组件上自定义的事件呢,可能你的第一反应是this.$emit


"input"
v-bind= "$attrs"
@blur= "blur">

{{errorTip}}

v-model= "input"
:errorTip= "errorTip"
@blur= "handleBlur">

el-input组件有4个自定义事件,还不算多,假如遇到自定义事件更多的第三方组件,要怎么办,难道一个一个添加进去,不仅会增加一堆非必要的methods,而且可读性差很容易和myInput自身的methods混在一起。其实可以用$listeners一步到位,先来看一下$listeners的定义。

$listeners:包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件。


"input"
v-bind= "$attrs"
v-on= "$listeners">

{{errorTip}}

v-model= "input"
:errorTip= "errorTip"
@blur= "handleBlur">

在myInput组件中只要在el-input组件上添加v-on="$listeners",就可以在myInput组件上使用el-input组件自定义的事件。


逆锋起笔是一个专注于程序员圈子的技术平台,你可以收获最新技术动态、最新内测资格、BAT等大厂的经验、精品学习资料、职业路线、副业思维,微信搜索逆锋起笔关注!

点个在看支持我吧,转发就更好了

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
马筱梅前夫彻底疯了!放出隐私照疑报复马筱梅,太扎汪小菲的心了

马筱梅前夫彻底疯了!放出隐私照疑报复马筱梅,太扎汪小菲的心了

小徐讲八卦
2024-06-05 06:56:16
中国人不可破的15条天规,太多人后悔知道得太晚!

中国人不可破的15条天规,太多人后悔知道得太晚!

流逝的沙
2024-06-03 12:19:31
A股要破3000点,6家国企财务造假,几十万散户被割,重罚60万元

A股要破3000点,6家国企财务造假,几十万散户被割,重罚60万元

鹏哥投研
2024-06-05 19:41:51
外交部:希望有关方保持冷静克制,谨言慎行

外交部:希望有关方保持冷静克制,谨言慎行

新京报
2024-06-05 19:19:31
交个朋友:618销售额已达去年同期九成,希望老罗能持续做直播

交个朋友:618销售额已达去年同期九成,希望老罗能持续做直播

澎湃新闻
2024-06-04 18:20:27
僵住了!荷媒:滕哈赫与曼联间陷入静默状态,双方都有顾虑

僵住了!荷媒:滕哈赫与曼联间陷入静默状态,双方都有顾虑

直播吧
2024-06-06 02:45:12
2名失联驴友去世后续:曝遇难女子是退休老师,退休工资9000元

2名失联驴友去世后续:曝遇难女子是退休老师,退休工资9000元

180°视角
2024-06-04 18:47:38
一张嘴毁了一部戏!韩式演技能不能滚出内娱?

一张嘴毁了一部戏!韩式演技能不能滚出内娱?

小片片说大片
2024-06-04 18:20:14
仅一夜!勇士老板神操作补强全明星球员,正式开启复仇之路!

仅一夜!勇士老板神操作补强全明星球员,正式开启复仇之路!

苏志vlog
2024-06-05 16:53:01
疑孟贺直播间再曝猛料:张兰曾想撮合汪小菲和章子怡,网友懵圈了

疑孟贺直播间再曝猛料:张兰曾想撮合汪小菲和章子怡,网友懵圈了

小徐讲八卦
2024-06-04 09:49:43
GDP增速跌破2%?

GDP增速跌破2%?

边际财经实验室
2024-06-05 08:17:07
2024年养老金调整有些“意外惊喜”,企退人员上涨150元难吗?

2024年养老金调整有些“意外惊喜”,企退人员上涨150元难吗?

社保小达人
2024-06-05 11:56:01
砖家:适当惩罚不生孩子的年轻人

砖家:适当惩罚不生孩子的年轻人

赢出个未来
2024-06-04 12:18:23
女子吐槽二婚老公对自己儿子不管不顾,生病都不问:二婚真没意思

女子吐槽二婚老公对自己儿子不管不顾,生病都不问:二婚真没意思

看晓天下事
2024-06-04 19:10:12
中国拒绝,俄罗斯拒绝,印度拒绝,不到24小时,泽连斯基对华摊牌

中国拒绝,俄罗斯拒绝,印度拒绝,不到24小时,泽连斯基对华摊牌

DS北风
2024-06-05 17:14:09
网友为什么对张镇麟和王芳这么大的恶意?杨毅:根源还是成绩不行

网友为什么对张镇麟和王芳这么大的恶意?杨毅:根源还是成绩不行

狼叔评论
2024-06-05 17:37:11
一干部接受纪律审查和监察调查

一干部接受纪律审查和监察调查

锡望
2024-06-05 17:06:29
闹剧结束了!王阳发声回应,获《追风者》剧组维护,王一博惨败

闹剧结束了!王阳发声回应,获《追风者》剧组维护,王一博惨败

猪猪侃娱乐
2024-06-05 18:27:42
全面封锁开始,中方军舰全撤走,马科斯反应微妙,菲船物资已抛海

全面封锁开始,中方军舰全撤走,马科斯反应微妙,菲船物资已抛海

说天说地说实事
2024-06-05 20:49:30
超越凯恩、姆巴佩和阿圭罗,哈兰德职业生涯已22次上演帽子戏法

超越凯恩、姆巴佩和阿圭罗,哈兰德职业生涯已22次上演帽子戏法

直播吧
2024-06-06 03:17:05
2024-06-06 03:28:49
技术小生
技术小生
互联网技术与技术人的职业发展
941文章数 497关注度
往期回顾 全部

科技要闻

9家车企拿到自动驾驶落地入场券,没特斯拉

头条要闻

9旬老人去世后被发现与小38岁保姆结婚 房产被过户

头条要闻

9旬老人去世后被发现与小38岁保姆结婚 房产被过户

体育要闻

赴美试训的崔永熙,表现究竟怎么样?

娱乐要闻

《青春有你》胡文煊被曝孕期出轨

财经要闻

员工证实周大福关闭深圳工厂

汽车要闻

又一个水桶车 试驾新“卷王”极狐阿尔法S5

态度原创

艺术
亲子
旅游
公开课
军事航空

艺术要闻

穿越时空的艺术:《马可·波罗》AI沉浸影片探索人类文明

亲子要闻

怎么给宝宝脱掉纸尿裤,这三个方法可以帮到你

旅游要闻

一女子从家到机场仅10分钟却没赶上飞机 愤而投诉

公开课

近视只是视力差?小心并发症

军事要闻

乌方:俄乌冲突升级 所有“红线”不复存在

无障碍浏览 进入关怀版