首先明确一下,这里所说的系统模块划分,是针对client,service,common这样的技术划分,而不是针对具体业务的模块划分。避免由于歧义,造成你的时间浪费。
公司内部某技术团队,在引用我们系统的client包时,启动失败。
失败原因是由于client下有一个cache相关的依赖,其注入失败导致的。
然后,就发出了这样一个疑问:我只是希望使用一个hsf接口,为什么还要引入诸如缓存,web处理工具等不相关的东西。
这也就自然地引出了前辈对我的一句教导:对外的client需要尽可能地轻便。
很明显,我们原有的client太重了,包含了对外的rpc接口,相关模型(如xxxdto),工具包等等。
可能有人就要提出,直接rpc+模型一个包,其它内容一个包不就ok了嘛?
问题真的就这么简单嘛?
其实出现上述问题,是因为在系统设计之初,并没有深入思考client包的定位,以及日后可能遇到的情况。
这也就导致了今天这样的局面,所幸目前的外部引用并不多,更多是内部引用。及时调整,推广新的依赖,与相关规范为时不晚。
先说说我以前的模块拆分。最早的拆分是每个业务模块主要拆分为:
这种拆分方式,是我早期从一份微服务教程中看到的。优点是简单明了,调用方可选择性地选择需要的模块引入。
至于一些通用的组件,如统一返回格式(如serverresponse,rtobject),则放在了最早的核心(功能核心,但内容很少)模块上。
后来,认为这样并不合适,不应该将通用组件放在一个业务模块上。所以建立了一个base模块,用来将通用的组件,如工具,统一返回格式等都放入其中。
另外,将每个服务都有的xxx-common模块给取消了。将其中的模型,放入了xxx-client,毕竟是外部调用需要的。将其中的工具,根据需要进行拆分:
上述这个方案,也就是我在负责某物联网项目时采用的最终模块划分方式。
在当时的业务下,该方案的优点是模块清晰,较为简洁,并且尽可能满足了迪米特原则(可以参考《阿里java开发手册相关实践》)。缺点则是需要一定的技术水平,对组件的功能域认识清晰。并且需要有一定的设计思考与能力(如上述工具拆分的第三点-xxx-client,明白为什么这是设计缺陷导致,并能够解决)。
那么,既然上述的方案挺不错的,为什么不复用到现在的项目呢?
因为业务变了,导致应用场景变了。而这也带来了新的问题,新的考虑角度。
原先的物联网业务规模并不大,所以依赖也较为简单,也并不需要进行依赖的封装等,所以针对主要是client的内/外这一维度考虑的。
但是现有的业务场景,由于规模较大,模块依赖层级较多,导致上层模块引入过多的依赖。如有一个缓存模块,依赖tair-starter(一个封装的key-value的存储),然后日志模块依赖该缓存模块(进行性能优化),紧接着日志模块作为一个通用模块,被放入了common模块中。依赖链路如下:
调用方 -> common模块 -> 日志模块 -> 缓存模块 -> tair-starter依赖
但是有的调用方表示,根本就不需要日志模块,却引入了tair-starter这一重依赖(starter作为封装,都比较重),甚至由于tair-starter的内部依赖与自身原有依赖冲突,得去排查依赖,进行exclude。
但是同时,也有的调用方,系统通过rich客户端,达到性能优化等目标。
所以,现有的业务场景除了需要考虑client的内/外这一维度,还需要考虑client的pool/rich这一维度。
可能有的小伙伴,看到这里有点晕乎乎的,这两个维度考量的核心在哪里?
内/外,考虑的是按照内外这条线,尽量将client设计得简洁,避免给调用方引入无用依赖。
而pool/rich,考虑的是性能,用户的使用成本(是否开箱即用)等。
最终的解决方案是对外提供3+n
这个方案,换个思路,理解也简单。
我们提供相关的能力,具体如何选择,交给调用方决定。
其实,讨论中还提到了bom方案(通过dependentmanagement进行jar包版本管理)。不过分析后,我们认为bom方案更适合那些依赖集比较稳定的client,如一些中间件。而我们目前的业务系统,还在快速发展,所以并不适用。
简单来说,直接从用户需求考虑(这里的用户就是调用方):
这里补充一点,我对讨论中一个问题的回答,这里提一下。
有人提到工具类,应该如何划分。因为有的工具类,是不依赖于服务状态的,如cookieutil进行cookie处理。有的工具类,是依赖于服务状态的,如redisutil包含redispool状态,直连redis,处理redis请求与响应。
其实这里有两个细节:
愿与诸君共进步。
如对本文有疑问, 点击进行留言回复!!
14、Ribbon整合断路器监控Hystrix Dashboard
网友评论