一、前言
乱码七糟 [luàn qī bā zāo],我时常怀疑这个成语是来形容程序猿的! 无论承接什么样的需求,是不是身边总有那么几个人代码写的烂,但是却时常有测试小姐姐过来聊天(_求改bug_)、有产品小伙伴送吃的(_求写需求_)、有业务小妹妹陪着改代码(_求上线_),直至领导都认为他的工作很重要,而在旁边的你只能蹭点吃的。
那你说,CRUD的代码还想让我怎么样?
这样的小伙伴,可能把代码写的很直接,ifelse多用一点,满足于先临时支持一下,想着这也没什么的。而且这样的业务需求要的急又都是增删改查的内容,实在不想做设计。而如果有人提到说好好设计下,可能也会被反对不要过度设计。
贴膏药似的修修补补,一次比一次恐怖!
第一次完成产品需求实在是很快,但互联网的代码不比传统企业。在传统行业可能一套代码能用十年,但在互联网高速的迭代下你的工程,一年就要变动几十次。如果从一开始就想着只要完成功能就可以,那么随之而来的是后续的需求难以承接,每次看着成片成片的代码,实在不知如何下手。
在研发流程规范下执行,才能写出好程序!
一个项目的上线往往要经历业务需求、产品设计、研发实现、测试验证、上线部署到正式开量,而这其中对研发非常重要的一环就是研发实现的过程,又可以包括为;架构选型、功能设计、设计评审、代码实现、代码评审、单测覆盖率检查、编写文档、提交测试。所以在一些流程规范下,其实很难让你随意开发代码。
开发代码的过程不是炫技,就像盖房子如果不按照图纸来修建,回首就在山墙上搭一个厨房卫浴!可能在现实场景中这很荒唐,但在功能开发中却总有这样的代码。
**所以我们也需要一些设计模式的标准思想,去建设代码结构,提升全局把控能力。**
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#%E4%BA%8C%E3%80%81%E5%BC%80%E5%8F%91%E7%8E%AF%E5%A2%83)二、开发环境
JDK 1.8
Idea + Maven
涉及工程三个,可以通过关注**公众号**:[bugstack虫洞栈(opens new window)](https://bugstack.cn/assets/images/qrcode.png),回复源码下载获取(打开获取的链接,找到序号18)
工程
描述
itstack-demo-design-3-00
场景模拟工程,模拟装修过程中的套餐选择(豪华、田园、简约)
itstack-demo-design-3-01
使用一坨代码实现业务需求,也是对ifelse的使用
itstack-demo-design-3-02
通过设计模式优化改造代码,产生对比性从而学习
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#%E4%B8%89%E3%80%81%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E4%BB%8B%E7%BB%8D)三、建造者模式介绍

图片来自:[https://refactoringguru.cn/design-patterns/builder(opens new window)](https://refactoringguru.cn/design-patterns/builder)
**建造者模式所完成的内容就是通过将多个简单对象通过一步步的组装构建出一个复杂对象的过程。**
那么,哪里有这样的场景呢?
例如你玩王者荣耀的时的初始化界面;有三条路、有树木、有野怪、有守卫塔等等,甚至依赖于你的网络情况会控制清晰度。而当你换一个场景进行其他不同模式的选择时,同样会建设道路、树木、野怪等等,但是他们的摆放和大小都有不同。这里就可以用到建造者模式来初始化游戏元素。
而这样的根据相同的物料,不同的组装所产生出的具体的内容,就是建造者模式的最终意图,也就是;**将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。**
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#%E5%9B%9B%E3%80%81%E6%A1%88%E4%BE%8B%E5%9C%BA%E6%99%AF%E6%A8%A1%E6%8B%9F)四、案例场景模拟

**这里我们模拟装修公司对于设计出一些套餐装修服务的场景。**
很多装修公司都会给出自家的套餐服务,一般有;欧式豪华、轻奢田园、现代简约等等,而这些套餐的后面是不同的商品的组合。例如;一级&二级吊顶、多乐士涂料、圣象地板、马可波罗地砖等等,按照不同的套餐的价格选取不同的品牌组合,最终再按照装修面积给出一个整体的报价。
这里我们就模拟装修公司想推出一些套餐装修服务,按照不同的价格设定品牌选择组合,以达到使用建造者模式的过程。
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_1-%E5%9C%BA%E6%99%AF%E6%A8%A1%E6%8B%9F%E5%B7%A5%E7%A8%8B)1. 场景模拟工程
itstack-demo-design-3-00
└── src└── main└── java└── org.itstack.demo.design├── ceilling│ ├── LevelOneCeiling.java│ └── LevelTwoCeiling.java├── coat│ ├── DuluxCoat.java│ └── LiBangCoat.java│ └── LevelTwoCeiling.java├── floor│ ├── DerFloor.java│ └── ShengXiangFloor.java├── tile│ ├── DongPengTile.java│ └── MarcoPoloTile.java└── Matter.java@小傅哥: 代码已经复制到剪贴板
在模拟工程中提供了装修中所需要的物料;ceilling(吊顶)、coat(涂料)、floor(地板)、tile(地砖),这么四项内容。(_实际的装修物料要比这个多的多_)
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_2-%E5%9C%BA%E6%99%AF%E7%AE%80%E8%BF%B0)2. 场景简述
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_2-1-%E7%89%A9%E6%96%99%E6%8E%A5%E5%8F%A3)2.1 物料接口
public interface Matter {String scene(); // 场景;地板、地砖、涂料、吊顶String brand(); // 品牌String model(); // 型号BigDecimal price(); // 价格String desc(); // 描述}@小傅哥: 代码已经复制到剪贴板
物料接口提供了基本的信息,以保证所有的装修材料都可以按照统一标准进行获取。
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_2-2-%E5%90%8A%E9%A1%B6-ceiling)2.2 吊顶(ceiling)
**一级顶**
public class LevelOneCeiling implements Matter {public String scene() {return "吊顶";}public String brand() {return "装修公司自带";}public String model() {return "一级顶";}public BigDecimal price() {return new BigDecimal(260);}public String desc() {return "造型只做低一级,只有一个层次的吊顶,一般离顶120-150mm";}
}
@小傅哥: 代码已经复制到剪贴板
`**二级顶**`
public class LevelTwoCeiling implements Matter {
public String scene() {return "吊顶";
}public String brand() {return "装修公司自带";
}public String model() {return "二级顶";
}public BigDecimal price() {return new BigDecimal(850);
}public String desc() {return "两个层次的吊顶,二级吊顶高度一般就往下吊20cm,要是层高很高,也可增加每级的厚度";
}
}
@小傅哥: 代码已经复制到剪贴板
`1`#### `[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_2-3-%E6%B6%82%E6%96%99-coat)2.3 涂料(coat)`
`**多乐士
23
**立邦地板(floor)``**德尔**`
public class DerFloor implements Matter {
public String scene() {return "地板";
}public String brand() {return "德尔(Der)";
}public String model() {return "A+";
}public BigDecimal price() {return new BigDecimal(119);
}public String desc() {return "DER德尔集团是全球领先的专业木地板制造商,北京2008年奥运会家装和公装地板供应商";
}
}
@小傅哥: 代码已经复制到剪贴板
`**圣象**`
public class ShengXiangFloor implements Matter {
public String scene() {return "地板";
}public String brand() {return "圣象";
}public String model() {return "一级";
}public BigDecimal price() {return new BigDecimal(318);
}public String desc() {return "圣象地板是中国地板行业著名品牌。圣象地板拥有中国驰名商标、中国名牌、国家免检、中国环境标志认证等多项荣誉。";
}
}
@小傅哥: 代码已经复制到剪贴板
#### `[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_2-5-%E5%9C%B0%E7%A0%96-tile)2.5 地砖(tile)`
`**东鹏**`
public class DongPengTile implements Matter {
public String scene() {return "地砖";
}public String brand() {return "东鹏瓷砖";
}public String model() {return "10001";
}public BigDecimal price() {return new BigDecimal(102);
}public String desc() {return "东鹏瓷砖以品质铸就品牌,科技推动品牌,口碑传播品牌为宗旨,2014年品牌价值132.35亿元,位列建陶行业榜首。";
}
}
@小傅哥: 代码已经复制到剪贴板
`**马可波罗**`
public class MarcoPoloTile implements Matter {
public String scene() {return "地砖";
}public String brand() {return "马可波罗(MARCO POLO)";
}public String model() {return "缺省";
}public BigDecimal price() {return new BigDecimal(140);
}public String desc() {return "“马可波罗”品牌诞生于1996年,作为国内最早品牌化的建陶品牌,以“文化陶瓷”占领市场,享有“仿古砖至尊”的美誉。";
}
}
@小傅哥: 代码已经复制到剪贴板
- `上就是本次装修公司所提供的装修配置单,接下我们会通过案例去使用不同的物料组合出不同的套餐服务。`
## `[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#%E4%BA%94%E3%80%81%E7%94%A8%E4%B8%80%E5%9D%A8%E5%9D%A8%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0)五、用一坨坨代码实现`
`讲道理没有ifelse解决不了的逻辑,不行就在加一行!`
`每一个章节中我们都会使用这样很直白的方式去把功能实现出来,在通过设计模式去优化完善。这样的代码结构也都是非常简单的,没有复杂的类关系结构,都是直来直去的代码。除了我们经常强调的这样的代码不能很好的扩展外,做一些例子demo工程还是可以的。`
### `[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#_1-%E5%B7%A5%E7%A8%8B%E7%BB%93%E6%9E%84)1. 工程结构`
itstack-demo-design-3-01 └── src └── main └── java └── org.itstack.demo.design └── DecorationPackageController.java
@小傅哥: 代码已经复制到剪贴板
``` 1
2
3
4
5
6
一个类几千行的代码你是否见过,嚯?那今天就让你见识一下有这样潜质的类!
**建造者模型结构**

工程中有三个核心类和一个测试类,核心类是建造者模式的具体实现。与ifelse实现方式相比,多出来了两个二外的类。具体功能如下;
Builder,建造者类具体的各种组装由此类实现。
DecorationPackageMenu,是IMenu接口的实现类,主要是承载建造过程中的填充器。相当于这是一套承载物料和创建者中间衔接的内容。
[#](https://bugstack.cn/md/develop/design-pattern/2020-05-26-%E9%87%8D%E5%AD%A6Java%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%E3%80%8A%E5%AE%9E%E6%88%98%E5%BB%BA%E9%80%A0%E8%80%85%E6%A8%A1%E5%BC%8F%E3%80%8B.html#%E4%B8%83%E3%80%81%E6%80%BB%E7%BB%93)七、总结
通过上面对建造者模式的使用,已经可以摸索出一点心得。那就是什么时候会选择这样的设计模式,当:一些基本物料不会变,而其组合经常变化的时候,就可以选择这样的设计模式来构建代码。
此设计模式满足了单一职责原则以及可复用的技术、建造者独立、易扩展、便于控制细节风险。但同时当出现特别多的物料以及很多的组合后,类的不断扩展也会造成难以维护的问题。但这种设计结构模型可以把重复的内容抽象到数据库中,按照需要配置。这样就可以减少代码中大量的重复。
设计模式能带给你的是一些思想,但在平时的开发中怎么样清晰的提炼出符合此思路的建造模块,是比较难的。需要经过一些锻炼和不断承接更多的项目,从而获得这部分经验。有的时候你的代码写的好,往往是倒逼的,复杂的业务频繁的变化,不断的挑战!
本文由博客一文多发平台 OpenWrite 发布!