“复杂要人命。它小莫开发者的生命,让产品难以规划、构建和测试。”—— Ray Ozzie
整洁的代码可以帮助开发团队在较低层的抽象层级上达成这一目标。
本节是关于如何保持较高层级——系统层级——上保持整洁。
将系统的构造和使用分开:
例子:酒店在建设时,使用起重机和升降机,而在酒店投入使用时,起重机和升降机就消失无踪,建筑物变得整洁。
软件系统应将启始过程和启始过程之后的运行时逻辑分离开,在启始过程中构建对象,也会存在互相缠结的依赖关系。
public Service getService(){
if (service == null){
service = new MyServiceImpl(...);
}
return service;
}
这是所谓的延迟初始化/赋值,好处:在真正使用对象之前,无需担心这种架空构造。坏处:必须要实现这个构造方法,否则无法编译,即这个对象无法被使用。
将构造和使用分开的方法:
(1)分解main,将系统中的全部构造过程搬迁到main或者main模块中:main函数创建对象,再将对象传递给应用程序,应用程序只管使用,对构造一无所知;
(2)如果应用程序需要负责确定何时创建对象,可以创建抽象工厂,让应用程序控制实体创建的时机;
(3)依赖注入,控制反转IoC是依赖管理的手段,它将应用需要的依赖对象的创建权责从对象中拿出来,放在一个专注于此事的对象中,并通过依赖注入(赋值器)将依赖对象传递给应用;
扩容:
面向方面编程(AOP),Java中三种方面和类似方面的机制:代理,纯AOP框架,AspectJ。
(1)java代理:适用于简单情况,如在单独对象或类中包装方法调用。代码量和复杂度是代理的两大弱点。
(2)纯Java AOP框架,如Spring AOP、JBoss AOP
(3)AspectJ:提供将方面作为模块构造处理支持的Java扩展
最佳系统架构由模块化的关注面领域组成,每个关注面均用纯Java(或其他语言)对象实现。不同领域之间用最不具有侵害性的方面或类方面工具整合起来。这种架构能测试驱动,就像代码一样。
拥有模块化关注面的POJO系统提供的敏捷能力,允许我们基于最新的知识做出优化的、时机刚好的决策。决策的复杂性也降低了。
领域特定语言(DSL)允许所有抽象层级和应用程序中的所有领域,从高级策略到底层细节,使用POJO来表达。