大话设计模式笔记(十)の建造者模式
举个栗子
问题描述
画一个小人,有头、身体、两手、两脚就可以了。
简单实现
人类
1. /**
2. * 人类
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public class Person {
6.
7. public void drawHead() {
8. System.out.print("头 ");
9. }
10.
11. public void drawBody() {
12. System.out.print("身体 ");
13. }
14.
15. public void drawHand() {
16. System.out.print("两手 ");
17. }
18.
19. public void drawLeg() {
20. System.out.print("两脚 ");
21. }
22.
23. }
24.
测试
1. public class Test {
2.
3. public static void main(String[] args) {
4. Person person = new Person();
5. person.drawHead();
6. person.drawBody();
7. person.drawHand();
8. person.drawLeg();
9. }
10.
11. }
12.
测试结果
1. 头 身体 两手 两脚
2.
存在问题
画人的时候,头身手脚是必不可少的,不管什么人物,开发时是不能少的。但上面测试代码中时各部分堆积起来,很容易漏写,比如导致健全的人物却少了一条“腿”。而且如果需要在别的地方用这些画小人的程序怎么办?
简单实现2
瘦人类
1. /**
2. * 瘦人类
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public class PersonThinBuilder {
6.
7. private Person person;
8.
9. public PersonThinBuilder(Person person){
10. this.person = person;
11. System.out.println("瘦人类:");
12. }
13.
14. // 建造瘦子
15. public void build(){
16. person.drawHead();
17. person.drawBody();
18. person.drawHand();
19. person.drawLeg();
20. }
21.
22. }
23.
胖人类
1. /**
2. * 胖人类
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public class PersonFatBuilder {
6. // 代码与瘦人类类似
7. }
8.
测试
1. public class Test2 {
2.
3. public static void main(String[] args) {
4. Person person = new Person();
5.
6. PersonThinBuilder thin = new PersonThinBuilder(person);
7. thin.build();
8.
9. System.out.println();
10. PersonFatBuilder fat = new PersonFatBuilder(person);
11. fat.build();
12. }
13.
14. }
15.
测试结果
1. 瘦人类:
2. 头 身体 两手 两脚
3. 胖人类:
4. 头 身体 两手 两脚
5.
存在问题
这样写的确达到了可以复用这两个画小人的目的,但容易“缺胳膊少腿”的问题,依然没有解决,比如现在需要加一个高个的小人,同样会因为编程不注意,产生同样问题。
建造者模式
定义
将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。 如果使用建造者模式,那么用户就只需指定需要建造的类型就可以得到它们,而具体建造的过程和细节就不需要知道了。
UML图
代码实现
建造人抽象类
1. /**
2. * 建造人抽象类
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public abstract class PersonBuilder {
6.
7. protected Person person;
8.
9. public PersonBuilder(Person person){
10. this.person = person;
11. }
12.
13. public abstract void buildHead();
14. public abstract void buildBody();
15. public abstract void buildArmLeft();
16. public abstract void buildArmRight();
17. public abstract void buildLegLeft();
18. public abstract void buildLegRight();
19.
20. }
21.
瘦人类2
1. /**
2. * 瘦人类2
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public class PersonThinBuilder extends PersonBuilder {
6.
7. public PersonThinBuilder(Person person){
8. super(person);
9. }
10.
11. @Override
12. public void buildHead() {
13. System.out.print("头 ");
14. }
15.
16. @Override
17. public void buildBody() {
18. System.out.print("身体 ");
19. }
20.
21. @Override
22. public void buildArmLeft() {
23. System.out.print("左手 ");
24. }
25.
26. @Override
27. public void buildArmRight() {
28. System.out.print("右手 ");
29. }
30.
31. @Override
32. public void buildLegLeft() {
33. System.out.print("左脚 ");
34. }
35.
36. @Override
37. public void buildLegRight() {
38. System.out.print("右脚 ");
39. }
40.
41. }
42.
胖人类2
1. /**
2. * 胖人类2
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public class PersonFatBuilder extends PersonBuilder{
6. // 代码与瘦人类2类似
7. }
8.
建造人指挥者
1. /**
2. * 建造人指挥者
3. * Created by callmeDevil on 2019/7/21.
4. */
5. public class PersonDirector {
6.
7. private PersonBuilder builder;
8.
9. public PersonDirector(PersonBuilder builder){
10. // 用户告诉指挥者需要什么样的小人
11. this.builder = builder;
12. }
13.
14. // 根据用户的选择建造小人
15. public void createPerson(){
16. builder.buildHead();
17. builder.buildBody();
18. builder.buildArmLeft();
19. builder.buildArmRight();
20. builder.buildLegLeft();
21. builder.buildLegRight();
22. }
23.
24. }
25.
测试
1. public class Test {
2.
3. public static void main(String[] args) {
4. Person person = new Person();
5. PersonDirector thinDirector = new PersonDirector(new PersonThinBuilder(person));
6. thinDirector.createPerson();
7.
8. System.out.println();
9. PersonDirector fatDirector = new PersonDirector(new PersonFatBuilder(person));
10. fatDirector.createPerson();
11. }
12.
13. }
14.
测试结果
1. 瘦人类2:
2. 头 身体 左手 右手 左脚 右脚
3. 胖人类2:
4. 头 身体 左手 右手 左脚 右脚
5.
此时如果需要增加一个高个子和矮个子的小人,应该怎么做?
加两个类,一个高个子类和一个矮个子类,都继承 PersonBuilder ,然后客户端调用即可。
总结
- 主要是用于创建一些复杂的对象,这些对象内部构建间的建造顺序通常是稳定的,但对象内部的构建通常面临着复杂的变化
- 建造者模式的好处就是使得建造代码与表示代码分离,由于建造者隐藏了该产品是如何组装的,所以若需要改变一个产品的内部表示,只需要再定义一个具体的建造者就可以了
- 建造者模式是在当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时适用的模式。
Pass:以上纯属个人理解~~如果发现有错或是心存建议意见等,欢迎大家评论或联系~(# ゚Д゚)~祝大家身体健康学习进步工作顺利生活愉快!
版权归 callmeDevil 所有,如需转载请标注转载来源
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 智能桌面机器人:用.NET IoT库控制舵机并多方法播放表情
· Linux glibc自带哈希表的用例及性能测试
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 手把手教你在本地部署DeepSeek R1,搭建web-ui ,建议收藏!
· 新年开篇:在本地部署DeepSeek大模型实现联网增强的AI应用
· Janus Pro:DeepSeek 开源革新,多模态 AI 的未来
· 互联网不景气了那就玩玩嵌入式吧,用纯.NET开发并制作一个智能桌面机器人(三):用.NET IoT库
· 【非技术】说说2024年我都干了些啥