【摘要】示例代码托管在:http://www.github.com/dashnowords/blogs

分享一篇尤大大演讲镇楼:「2019 JSConf.Asia - 尤雨溪」在框架设计中寻求平衡

一.框架的定位

框架通常只是一种设计模式的实现,它并不意味着你可以在开发中避免所有分层设计工作。

SPA框架几乎都是基于MVCMVVM设计模式而建立起来的,这些模式都只是宏观的分层设计,当代码量开始随着项目增大而增多时,问题就会越来越多。许多企业内部的项目仍然在使用angularjs1.X,你会发现许多controller的体积大到令人发指,稍有经验的团队会利用好angularjs1构建的controller,service,filter以及路由和消息机制来完成基本的拆分和解耦,这已经能让他们的开发能力中等体量的项目,往往只有掌握了angularjs1玩法精髓——directive的队伍,才能够在应付大型项目时使代码保持足够的清晰度,当然这只是在代码形态和模块划分上的工作,相当于代码的骨骼,想要让业务逻辑本身更加清晰,就需要更高级的建模设计知识来对业务逻辑进行分层,例如领域驱动模型。如果你仍然在使用angularjs1.x的版本进行开发,可以参考【如何重构Controller】进行基本的分层拆分设计。

有趣的是一些团队认为无法承载大型项目是angularjs1.x的原罪,与他们的开发水平无关,于是将希望寄托于拥有自动化工具加持的现代化SPA框架,然而如果有机会观察你就会发现,许多项目对新框架的使用方式和之前并没有本质的差别,只不过是把以前臃肿到不行的代码又换了一种形式塞进了前端工程里,然后借着ES6语法和新型框架本身的简洁性,开始沾沾自喜地认为这是自己重构的功劳。

请记住,如果不进行结构设计,即便使用最新版本的最热门的框架,写出来的代码依旧会是一团乱麻。

二. Vue开发中的script拆分优化

Vue框架为例,在工程化工具和vue-loader的支撑下,主流的开发模式是基于*.vue这种单文件组件形态的。一个典型的vue组件包含如下几个部分:

<template><!--视图模板-->
</template><script>/*编写组件脚本*/export default {name:'component1'}
</script><style>/*编写组件样式*/
</style>

script的部分通常包含有交互逻辑业务逻辑数据转换以及DOM操作,如果不加整理,很容易变得混乱不堪。*.vue文件的本质是View层代码,它应该尽可能轻量并包含与视图有关的信息,即特性声明事件分发,其他的代码理论上都应该剥离出去,这样当项目体量增大后,维护起来就更容易聚焦关键信息,下面就如何进行脚本代码拆分提供一些思路,有一些可能是很基本的原则,为尽可能完整就放在一起,你并不需要从最开始就采纳所有的建议。

1.组件划分

这是View层减重的基础,将可共用的视图组件剥离出去,改为消息机制进行通信,甚至直接剥离出包含视图和业务代码的业务逻辑组件,都可以有效地拆分View层,降低代码的复杂度。

2.剥离业务逻辑代码

script中最大的一部分一般是业务逻辑,首先将业务逻辑代码剥离为独立的[name].business.js模块,这样做的直观好处就是减轻了View层,另一方面是解除了业务逻辑和页面之间的强绑定关系,如果其他页面也涉及到这块业务逻辑中的个别方法,就可以直接进行复用,最后就是当项目逐渐复杂,你决定引入vuex来进行状态管理时View层会相对更容易修改。

一段包含基本增删改查逻辑的组件大概是下面的样子:

<script>export default{name:'XXX',methods:{handleClickCreate(){},handleClickEdit(){},handleClickRefresh(){},handleClickDelete(){},sendCreate(){},sendEdit(){},sendGetAll(){},sendDelete(){}}}
</script>

简易的剥离方式是将交互逻辑保留在视图层,将业务逻辑部分代码放在另一个模块中,然后利用ES6扩展运算符将其加入到组件实例的方法中,如下所示:

<script>import OrderBusiness from './Order.business.js';export default{name:'XXX',methods:{...OrderBusiness,handleClickCreate(){},handleClickEdit(){},handleClickRefresh(){},handleClickDelete(){},}}
</script>

这种方式只是一种形态上的模块化拆分,并没有对业务逻辑本身进行梳理。另一种方式是构建独立的业务逻辑服务,保留在View层中的代码很容易转换为使用vuex时的编码风格:

<script>import OrderBusiness from './Order.business.js';export default{name:'XXX',methods:{handleClickCreate(){OrderBusiness.sendCreate();},handleClickEdit(){OrderBusiness.sendEdit();},handleClickRefresh(){OrderBusiness.sendGetAll();},handleClickDelete(){OrderBusiness.sendDelete();}}}
</script>

笔者的建议是,前面三个示例随着项目体量的增长可以实现渐进式的修改。

3. 剥离数据转换代码

在前后端分离的开发模式下,前端所需要的数据支持需要从后端请求获得,但请求来的原始数据通常都是无法直接使用的,甚至有可能引发代码出错,例如时间可能是以时间戳形式传过来的,或者你的代码需要取用某个对象属性时,后台同学却在该属性上挂了一个默认值NULL等,另一方面,开发过程中的接口改动是无法避免的,所以在代码结构的设计上,应该尽可能将可能变化的部分聚合起来。

比较实用的做法就是为每一个接口建立一个Transformer函数,从后台请求来的数据先经过Transformer函数变换为前台能够流通使用的数据结构,并在必要的属性上添加适当的默认值防止报错,你可以尽情地在此使用Lodash.js等函数工具来加工和重组自己需要的数据,即使最初后台传给你的数据不需要加工,也可以保留一个透传函数或是模块说明以提醒其他协作开发者在面对这种场景时采用类似的做法,它的功能就是为逻辑层提供直接可用的数据。当前端代码越来越重时,TransformerRequest部分可以很方便地移动到中间层。

4. 善用computed和filters处理数据展示

对原始数据的转换并不能覆盖所有场景,这就需要在定制展示的场景中利用computedfilters,它们都可以用来在不改变数据的情况下更改展示结果,例如将数据中的0或1转换为未完成已完成,或者是将时间戳和当前时间作比较后改为可读性更高的刚刚,1分钟前,1小时前,1天前等等,这些开发场景中是不能采用强行赋值来处理的,这是就可以使用计算属性computed或过滤器filters来处理,它们的区别是computed一般用于组件内部,不具有通用性,而filters一般用于可复用的场景,可以通过下面的形式来定义一个展示效果为首字母大写的全局过滤器:

Vue.filter('capitalize', function (value) {if (!value) return '';value = value.toString();return value.charAt(0).toUpperCase() + value.slice(1);
})

当项目中使用vuex来进行状态管理时,computed通常会等价替换为state中的getter

5. 使用directive处理DOM操作

尽管Vue提供了refs这个接口来实现在逻辑层直接操作DOM,但我们应当尽可能避免将复杂的DOM操作放在这里,有时候页面上DOM变化的场景较多,将每个变化都使用数据驱动的方式显然是不合理的,这时就需要用到指令特性directive,它常用来补充实现一些业务逻辑无关的DOM变化(业务逻辑相关的变化大都通过数据绑定进行了自动关联)。directive的基本用法可以直接参考【官方指南】,需要注意的是许多初级开发者都不太在意内存泄漏的问题,在directive的使用中需要格外注意这一点,通常我们会在bind事件钩子中绑定事件并使用属性持有这个监听函数,并在unbind钩子中解除对同一个监听函数的绑定,即使没有使用自定义指令,你也需要建立在必要时解绑监听器的编码习惯:

Vue.directive('clickoutside',{bind:function (el, binding){//定义监听器function handler(e) {if (el.contains(e.target)) {return false;}if (binding.expression){binding.value(e);}}el.__clickOutSide__ = handler;document.addEventListener('click', handler);},unbind:function (el) {document.removeEventListener('click',el.__clickOutSide__);delete el.__clickOutSide__ ;}});

demo中提供了一个简单的directive示例,你可以用它来做练习。

demo.rar

md原文.rar

作者:大史不说话

Vue中拆分视图层代码的5点建议相关推荐

  1. 如何在Vue中添加百度统计代码?

    目录 第一步:在百度统计网站中添加自己的网站 第二步:在maim.js下添加百度统计代码 第三步:创建beforeEach方法 第四步:代码安装检查 第一步:在百度统计网站中添加自己的网站 百度统计官 ...

  2. Vue中使用Monaco Editor代码编辑器

    一.安装依赖 npm install editor@1.0.0 npm install monaco-editor@0.19.3 npm install monaco-editor-webpack-p ...

  3. “约见”面试官系列之常见面试题之第七十五篇之vue中如何使当前css起作用(建议收藏)

    方法很简单,在组件中的style前面加上scoped就可以了,示例: 本面试题为前端常考面试题,后续有机会继续完善.我是歌谣,一个沉迷于故事的讲述者. 欢迎一起私信交流. "睡服" ...

  4. Vue中视图层的拆分

    一.组件的划分 思路: 将可共用的视图组件剥离出去,如常见的header.footer.nav等,这些剥离复用的组件甚至可以带有完整的业务逻辑,这种方法比较常用,具体不同业务有不同做法. 二.剥离业务 ...

  5. 基于Vue.js的企业级前端代码架构设计设想

    在前端架构设计这块也已经工作了一段时间,也翻遍了很多书籍,但是就目前来说笔者还是没有看过真正把前端架构讲好的书,加上现在前端技术的发展诞生了许多新的框架,如:vue.react.angular,这也越 ...

  6. java中表示层 控制层 业务逻辑层 数据访问层

    控制层(controller):的职能是负责读取视图表现层的数据,控制用户的输入,并调用业务层的方法: 业务层(service):需要根据系统的实际业务需求进行逻辑代码的编写,有些业务逻辑需要通过与数 ...

  7. vue中echarts使用案例:饼图(可直接使用)

    目录 功能.效果图: 代码实现 1.下载echarts包:终端运行 2.代码 3.运行项目:终端运行 4.可能遇到的错误 具体代码详解: 功能.效果图: 1.点击对应模块,放大并显示数据 2.点击下方 ...

  8. Ant design vue中实现动态更换主题色

    一.创建vue项目 可视化创建步骤:cmd调出窗口,输入vue ui,自动在浏览器上打开新窗口 输入文件名 ,文件名最好是英文 然后进行配置功能, 根据需求选择配置功能 这里最好选择2.0版本, 可以 ...

  9. vue中实现模糊搜索

    vue中实现模糊搜索, 代码如下: 首先在框架中写一个input输入框 <template><div>//v-model实现双向绑定<input type="t ...

最新文章

  1. TZOJ上的C语言作业答案,C语言编程练习
  2. 怎么让电脑运行速度变快_电脑运行慢怎么办 电脑运行慢解决方法【详解】
  3. 用servlet设计OA管理系统时遇到问题
  4. Task.CompletedTask和Task.Result小记
  5. nyoj91 阶乘之和
  6. 揭秘大流量场景下发布如「丝般顺滑」背后的原因
  7. idea升级2019.3后字体有的粗有的细
  8. ASP与SQL数据库连接及SQL常用命令使用方法
  9. pace.js – 加载进度条插件
  10. 永城2021高考成绩查询,永城中考成绩查询2021
  11. 微信只允许二级分销,否则封停账号,三级分销何去何从?
  12. 计算机系vlog,华为Mate40系列化身vlog神器,专业好用两手抓
  13. Master HA彻底解密
  14. 【Python成长之路】python资料共享
  15. 攻防世界crypto高手题之best_rsa
  16. 【客户服务】如何进行大客户管理
  17. php 读取指定路径照片,必应每日图片合集程序之php读取指定目录图片
  18. Flowable集成钉钉实现抄送发送消息
  19. 【自习任我行】任务跟踪3
  20. 应届生深圳求职全攻略

热门文章

  1. 什么是php的ast结构,什么是AST?Vue源码中AST语法树的解析
  2. linux对于文本的操作,Linux文本文件操作
  3. 两个运放制作加法器_集成电路的分类及其制作工艺
  4. ogg 登录mysql报字符集_mysqldump之字符集问题解决
  5. 构造函数能默认初始化其静态成员么?
  6. npm包管理器小节一下
  7. POJ 3744:Scout YYF I 概率DP+特征方程+快速幂
  8. Unity3D 5.x 简单实例 - 发射炮弹
  9. 技术员例会记要(一)
  10. leetcode 130 python