设计模式之建造者模式详解(附源代码)

建造者模式

建造者模式

建造者模式(Builder Pattern)使用多个简单的对象一步一步构建成一个复杂的对象。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
一个 Builder 类会一步一步构造最终的对象。该 Builder 类是独立于其他对象的。

介绍

意图:将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。
主要解决:主要解决在软件系统中,有时候面临着"一个复杂对象"的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定。
何时使用: 一些基本部件不会变,而其组合经常变化的时候。
如何解决:将变与不变分离开。
关键代码:建造者:创建和提供实例,导演:管理建造出来的实例的依赖关系。
角色:

  1. builder:为创建一个产品对象的各个部件指定抽象接口。
  2. ConcreteBuilder:实现Builder的接口以构造和装配该产品的各个部件,定义并明确它所创建的表示,并提供一个检索产品的接口。
  3. Director:构造一个使用Builder接口的对象。
  4. Product:表示被构造的复杂对象。ConcreteBuilder创建该产品的内部表示并定义它的装配过程,包含定义组成部件的类,包括将这些部件装配成最终产品的接口。

优点:

  1. 建造者独立,易扩展。
  2. 便于控制细节风险。

缺点:

  1. 产品必须有共同点,范围有限制。
  2. 如内部变化复杂,会有很多的建造类

使用场景:

  1. 需要生成的对象具有复杂的内部结构。
  2. 需要生成的对象内部属性本身相互依赖。
    注意事项:与工厂模式的区别是:建造者模式更加关注与零件装配的顺序

实现:

听说最近人工智能可以写作了,那么这里我们也写一个能自动造句的人工智障智能,叫做鹏酱。鹏酱可以造主谓宾结构的句子。代码如下:

一、建造变动较小较慢的组件

1.主语系列(subject)

/**
 * 
 * @author Thornhill
 * @see 主语的接口
 */
public interface Subject {
    public void showSbuject();
}

/**
 * 
 * @author Thornhill
 * @see 主语“做饭”
 */
public class Cook implements Subject {

    @Override
    public void showSbuject() {
        System.out.print("做饭");
    }

}

/**
 * 
 * @author Thornhill
 * @see 主语“编程”
 */
public class Coding implements Subject {

    @Override
    public void showSbuject() {
        System.out.print("编程");
    }

}

2.谓语系列(predicate)

/**
 * 
 * @author Thornhill
 * @see 谓语接口
 */
public interface Predicate {
    public void showPredicate();
}


/**
 * 
 * @author Thornhill
 * @see 谓语“是”
 */
public class Is implements Predicate {

    @Override
    public void showPredicate() {
        System.out.print("是");
    }

}

/**
 * 
 * @author Thornhill
 * @see 谓语“不是”
 */
public class IsNot implements Predicate {

    @Override
    public void showPredicate() {
        System.out.print("不是");
    }

}

3.宾语系列(object)

/**
 * 
 * @author Thornhill
 * @see "宾语接口"
 */
public interface Object {
    public void showObject();
}

/**
 * 
 * @author Thornhill
 * @see 宾语“简单”
 */
public class Easy implements Object {

    @Override
    public void showObject() {
        System.out.println("简单的");

    }

}

/**
 * 
 * @author Thornhill
 * @see 宾语“困难”
 */
public class Difficult implements Object {

    @Override
    public void showObject() {
        System.out.println("困难的");      
    }

}

二、建造不会变动的产品(Product)框架

/**
 * 
 * @author Thornhill
 * @see 产品框架(Product)
 */
public class ChickenSoup {
    private Subject subject;
    private Predicate predicate;
    private Object object;

    public Subject getSubject() {
        return subject;
    }

    public void setSubject(Subject subject) {
        this.subject = subject;
    }

    public Predicate getPredicate() {
        return predicate;
    }

    public void setPredicate(Predicate predicate) {
        this.predicate = predicate;
    }

    public Object getObject() {
        return object;
    }

    public void setObject(Object object) {
        this.object = object;
    }

    public void showChickenSoup() {
        subject.showSbuject();
        predicate.showPredicate();
        object.showObject();
    }
}

三、建造Builder接口:

/**
 * 
 * @author Thornhill
 * @see 鸡汤文章的接口(Builder)
 */
public interface EssayBuilder {

    public void setEssaySubject();

    public void setEssayPredicate();

    public void setEssayObject();

    public ChickenSoup getEssay();
}

四、建造实现Builder接口的具体生成器(ConcreteBuilder):

/**
 * 
 * @author Thornhill
 * @see 构造“做饭简单”的具体生成器
 */
public class CookingIsEasy implements EssayBuilder {

    private ChickenSoup chickenSoup = new ChickenSoup();

    @Override
    public void setEssaySubject() {
        chickenSoup.setSubject(new Cook());

    }

    @Override
    public void setEssayPredicate() {
        chickenSoup.setPredicate(new Is());
    }

    @Override
    public void setEssayObject() {
        chickenSoup.setObject(new Easy());

    }

    @Override
    public ChickenSoup getEssay() {
        return chickenSoup;
    }

}

/**
 * 
 * @author Thornhill
 * @see 构造“编程不难”的具体生成器
 */
public class CodingIsNotDiffcult implements EssayBuilder {

    private ChickenSoup chickenSoup = new ChickenSoup();

    @Override
    public void setEssaySubject() {
        chickenSoup.setSubject(new Coding());

    }

    @Override
    public void setEssayPredicate() {
        chickenSoup.setPredicate(new IsNot());

    }

    @Override
    public void setEssayObject() {
        chickenSoup.setObject(new Difficult());

    }

    @Override
    public ChickenSoup getEssay() {
        return chickenSoup;
    }

}

五、建造通过Builder接口,按照顺序调用ConcreteBuilder的导演(Director):

/**
 * 
 * @author Thornhill
 * @see 调用具体生成器的“导演”类
 */
public class EssayDirector {
    public ChickenSoup getFinalEssay(EssayBuilder eb) {
        eb.setEssayObject();
        eb.setEssayPredicate();
        eb.setEssaySubject();
        return eb.getEssay();
    }
}
六、测试:
/**
 * 
 * @author Thornhill
 * @see 建造者模式测试
 */
public class Test {

    public static void main(String[] args) {
        ChickenSoup essay1 = new EssayDirector().getFinalEssay(new CookingIsEasy());
        essay1.showChickenSoup();
        
        ChickenSoup essay2 = new EssayDirector().getFinalEssay(new CodingIsNotDiffcult());
        essay2.showChickenSoup();
        
    }

}

结果

做饭是简单的
编程不是困难的

结语:

建造者模式适用于组建不变而组合常有变动的情景。产品(Product),抽象接口(Builder),导演(Director)通常是不用变的部分;而具体生成器(ConcreteBuilder)是扩展时增加的部分,通常改动只改动具体生成器。

GitHub源代码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,589评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,615评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,933评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,976评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,999评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,775评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,474评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,359评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,854评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,007评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,146评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,826评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,484评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,029评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,153评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,420评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,107评论 2 356

推荐阅读更多精彩内容