整洁架构

如果你对整洁架构(Clean Architecture)有所了解的话,回想一下我们前几篇中描述的内容,你会发现整洁架构对前端,对 MVP 来说也是同样适用的。

关于什么是整洁架构完全可以通过阅读 Uncle Bob 原版图书中文版《整洁架构之道》来了解,或者可以通过阅读他的一个简短版本博客 The Clean Architecture 一探端倪。但我还是推荐阅读图书,图书全面而且浅显易懂,没有和某一门编程语言强行绑定,即使你没有后端背景也能流畅的通读下来。出于篇幅的考虑,在这里我只取一瓢,摘取一个契合于我们前端架构的知识点以做说明,就是模块间的依赖关系

这篇文章里更多的是告诉你 what(结论),而不是 why(为什么需要这么做)。因为 why 这件事需要更多的上下文来解释,这也是我为什么推荐你阅读原书的另一个原因。

为了便于说明,我也把上次的 MVP 流程也放在这里

依赖关系

我们可以将我们的应用大致划分为三类模块:UI、Service Layer (Presenter),Business Model。这么划分不仅是直觉上告诉我们应该怎么做,更重要的是这三大类的模块的变化频率是不同的,这也是我前文中说的 axis of change.

变化频率最高的是 UI。这里的变化频率不仅仅指的是我们实际上修改代码的次数,还包括它可以”被替换的程度"。什么意思呢?想象一下目前你正在编写的任何前端应用,关于视图的渲染可能使用的是 React, Angular, 甚至还可能只是 Node.js 的命令行而已。但无论视图层的框架是什么,这个应用的核心功能是不会变的,如果它是一个计算器,不同视图框架编写的不同之处无非在于,用户是通过 React 的输入框、 Angular 输入框还是电脑终端输入需要计算的数值。

变化频率次高的是 Service Layer,也就是用户用例,最后才是 Business Model。Service Layer 通常起的是编排作用,你也可以简单理解为流程控制,它通过调用各个 Business Model 来完成用户的一次交互,相信你完全可以理解流程的更改是比较频繁的。而至于 Business Model,则是不会轻易更改的,或者说修改起来是非常慎重的。想象税务系统或者移民系统中的每一个业务领域,每一个规则都和法律法规相关联,有的和公司的政策和盈利模式相关联,是不能随意修改的。

还是想强调我描述的其实是一种相对情况,你在实际过程中可能你的 Service Layer 和 Business Model 会有相同的修改频率。有时候这种情况是正常的,但有时候是则是危险的信号。

什么是危险的信号?

我们想象一段用计算星座的 React 组件代码:

function ZodiacComponent({ year, month, day }) {const result = zodiacService.get(year, month, day);return <div>{result}</div>
}

假设我们现在需要把UI重构,用 Angular 改写:

@Component({selector: 'app-zodiac',templateUrl: './app.component.html',styleUrls: ['./app.component.css']
})
export class AppComponent {@Input() year: number;@Input() month: number;@Input() day: number;result:numberconstructor(private zodiacService: ZodiacService) {this.result = this.zodiacService.get(year, month, day);}
}

不难看出无论是在 React 还是 Angular 中最核心的功能在于 zodiacService.get方法,它通过传入的年、月、日来计算用户的星座。但无论你用什么样的UI框架对应用视图进行重写,这部分的计算逻辑是不需要发生变化的。这是我们的核心业务逻辑。

业务不需要关心 UI。这是想当然的事情。

但如果你在重写应用时发现修改 UI 的同时也需要修改业务逻辑,这个可能就会有问题了。这么做是完全可能的,例如在 React 中你这么向 zodiacService 的 get方法传递的不是年月日,而是组件信息:

class ZodiacService {get(componentRef) {const year = componentRef.current.querySelector('#input-year').value;const month = componentRef.current.querySelector('#input-month').value;const day = componentRef.current.querySelector('#input-day').value;}
}

很显然,这样的get方法因为和 React 组件机制进行了强绑定,当我们需要使用 Angular 重构时,这一部分代码无法被 Angular 组件所使用。

这只是其中一种模块与UI产生关联的情况,在现实代码中,我们很多时候都会“一不小心”对UI产生了依赖。

UI与业务逻辑可能算是极端的情况,但是 Service Layer 和 Business Model 之间呢,Service Layer 和 UI 之间呢,相邻层之间更容易产生相互依赖和耦合。耦合在代码中的表现症状可能是 bad smell 中的 “散弹式修改”,也可能是“依恋情结”,“发散式变化”。但它们背后的原因根本原因其实是模块之间产生了不合理的依赖。

所以文章最开始的(圆环套圆环的)整洁架构图中,**不同层(圆环)之间的依赖关系是有方向性的,并且方向是指向内部的,即每一个圈的内部都对它的外部一无所知,这里的一无所知包括函数、变量等等。**所以我们能看到 Business Model 不知道调用它的 Service Layer 究竟完成的是什么样的流程,它也更不知道最外层的 UI 使用的是什么框架。

这个思路不仅对于前端是适用的,对于其他类型的应用也同样是适用的。只不过对于后端应用来说,产生的交互的,可以被替换的不仅是前端,还可是API。

实战

最为这个系列的最后一篇,在之前谈论了那么多理论之后,我想正面的回答一个很多人心中应该会有的问题:所有的组件、所有的应用,都必须按照 Flux、MVP、整洁架构来开发吗?

如果一定要按照是或者不是来回答的,答案一定是不是。

更准确的答案是:依照情况而定。

比如一个加载数据同时渲染在列表上的组件:

function UserList() {const [data, setData] = useState([])useEffect(() => {requestUserList().then(response => setData(response))}, []);return {data.map(item => <User data={item} />)}
}

这个组件可以再做拆分吗?当然可以;但是它还有必要拆分吗?我认为没有。

这又要回到我们第一篇中说的,本质上所有的设计,都是为了非功能需求。尤其是为了降低今后的维护成本。首先这段代码是极其简单的,这非常好,降低了我们系统的复杂性;其次如果你认为这段代码今后的维护成本在可控的范围之内,那么它其实就没有再进行拆分的必要。

那你可能又会问了,什么时候该拆分呢?我想回答:当它该拆分的时候。

这不是废话吗!

问题其实在于,我很难给出你一个精确的标准,比如当 XX 到达 XX 时,当 XX 遇到 XX 情况时就应该拆分;又或者说,拆分的标准太广了,Martin Fowler的《重构》或者是是 Uncle Bob 的 《整洁代码之道》都可以作为拆分的标准。

这个没有标准的标准,完全取决你的个人决策,恰恰也是最考验功力的地方,这也是最能体现你价值的地方。

服务推荐

  • 蜻蜓代理
  • ip代理服务器
  • 企业级代理ip
  • 微信域名检测
  • 微信域名拦截检测

前端架构 101(六):整洁(Clean Architecture)架构是归宿相关推荐

  1. 整洁架构读书笔记(Clean Architecture)

    又称干净的架构The Clean Architecture,这是著名软件工程大师Robert C Martin提出的一种架构整洁清晰之道,也是当前各种语言开发的目标架构.干净.清晰.整洁的架构应该只包 ...

  2. 音频文件 数据库存储_刚哥谈架构 (六) 谈谈数据库架构

    无论是构建什么样的应用,大都离不开数据.而在应用的架构设计中,如何设计数据库,使用什么类型的数据库,就是一个架构师必须了解的.所有的数据库的共同点都是以某种方式存储数据,以某种接口来访问存储的数据.我 ...

  3. Clean Architecture 实施指南

    在之前的那篇<整洁前端架构>的文章里, 我们介绍了如何在前端领域里使用 Clean Architecture.在过去的几个月里,我们实践了 Clean Architecture 架构,并且 ...

  4. 基于springboot+redis+bootstrap+mysql开发一套属于自己的分布式springcloud云权限架构(七)【权限架构生产者(数据字典)】

    在我们的编码中我们经常会存放一些静态常量,而这些静态变量我们通常都是写死的,如果我们需要改变这些静态变量的值那么我们必须要修改代码,并重新发布程序,这很明显很不符合我们现实中的业务场景,我们现实中的业 ...

  5. 前端切换视图_前端架构 101(五):从 Flux 进化到 Model-View-Presenter

    李熠:前端架构 101(一):在谈论它们之前我们需要达成的共识​zhuanlan.zhihu.com 李熠:前端架构 101(二): MVC 初探​zhuanlan.zhihu.com 李熠:前端架构 ...

  6. 前端 input怎么显示null_前端架构 101(二): MVC 初探

    我把 MVC 框架作为我们理解架构的切入点.虽然它现在已经式弱了,但在我看来它非常重要并且起到了承上启下的作用:作为经典的解决方案第一次系统的把应用的从复杂的混沌中解救了出来.从这套方法论中我们能学习 ...

  7. go 微服务框架_清晰架构(Clean Architecture)的Go微服务

    更新 我最进对这个框架进行了重大升级,虽然所做的改动不大,但成效显著.这次更新修复了旧框架中的所有主要问题,尽管它的的主要项目结构和接口都没有改变,具体改动内容请参见清晰架构(Clean Archit ...

  8. 瞎谈干净架构(clean architecture)

    写在前面的话:最近正在写一个自己的开源项目,打算使用像rxjava,retrofit,dagger2等开源框架,结果一个多月过去了,大框架基本写好了,却因为布局想绚丽一点,一直都没确定,最后拖延症犯了 ...

  9. 清晰架构(Clean Architecture)的Go微服务: 程序设计

    我使用Go和gRPC创建了一个微服务,并将程序设计和编程的最佳实践应用于该项目. 我写了一系列关于在项目工作中做出的设计决策和取舍的文章,此篇是关于程序设计. 程序的设计遵循清晰架构(Clean Ar ...

最新文章

  1. wordpress rest api 登录_WordPress版微信小程序3.5版发布
  2. SCA与spring集成(在spring中开发SOA)
  3. 【MRCTF—Web】做题+复现记录
  4. VTK:可编程字形过滤器用法实战
  5. outlook错配置账户到exchange后的删除办法
  6. 常用数据库连接串与驱动总结
  7. Swift 在 GAIA 平台云端一体化的探索
  8. python3l下载_lunix 安装python3
  9. windows查看GPU信息(nvidia-smi)
  10. ASP.NET中实现无刷新级联
  11. xcode 4.0 以上版本安装新浪微博SDK
  12. 【P01】双运放非反相放大器和失调处置
  13. 码农小汪之理解Java注解。
  14. 手工查杀威金病毒,完全恢复exe程序
  15. 2014年美国大学生数学建模竞赛翻译及建模思路
  16. 2012年8月11日
  17. 电脑和ubuntu开发板用网线连接的方法
  18. 【他山之石】康师傅的本地化策略
  19. 元学习 迁移学习_元学习就是您所需要的
  20. python数学库函数包括求最大公约数函数吗_pythonmathcot函数_math库是python语言的数学模块吗_Python中的数学模块:...

热门文章

  1. 【DSA】树-哈弗曼树详解(3)
  2. Servlet 和Session的关系
  3. 基于BP 网络分类器的交通标志识别
  4. php快速计算子网掩码的前缀长度
  5. 北大博士把“计算机底层原理“讲的如此通俗易懂
  6. 网校系统搭建服务有哪些优势?为什么要试水网校系统开发?
  7. 分享如何利用思维导图进行营销策划使用攻略
  8. RSI指标使用技巧和参数设置
  9. Java学习——传说中的13个规范
  10. petalinux uboot源码怎么打补丁