zoukankan      html  css  js  c++  java
  • 《大话设计模式》——建造者模式

    建造者模式结构图

    从上图中可以看到实现建造者模式至少需要4个角色:

    1、director:指挥者。由指挥者根据需求决定调用哪个建造者生产哪种产品。

    1、builder:建造者接口。定义了建造一个产品所必须的所有流程。所有的建造者都需要实现该类,以确保实现建造一个产品所需的流程。

    2、concreteBuilder:具体建造者。每一个建造者,可以创建一种产品,产品细节在建造者中定义,因此多个产品对应多个建造者。

    3、product:具体产品。

    我们这里以厨师炒菜来举例:

    假如有一家大排档,它只有一个产品:炒面,目前有两个菜单,菜单上面严格要求了炒面时放的盐多少克,油多少毫升等等,这两个菜单一个是蛋炒面,一个是肉炒面,现在来了一个客人点了一碗蛋炒面,过了不久又来了一个客人点了一碗肉炒面。厨师就分别拿着这两个菜单去炒面。

    那么在这里:product是炒面,director是厨师,builder是炒面的过程,concreteBuilder是两个菜单。

    1、先定义炒面product

    public class Product{
        List<String> chaofen = new List<String>();
    
        public void add(String part){
            chaofen.add(part); //炒面里面可以添加蛋或者肉或者其他的东西
        }
    
    }

    2、然后定义炒面的步骤builder,分为两步,1添加面,2添加蛋或肉。还有一些细节油盐等我们这里为了简单就省略掉了。

    inteface Builder{
        void addNoodles(); //炒粉有两个步骤,这里添加面
        void addEggOrMeat(); //炒粉有两个步骤,这里添加蛋或者肉
        Product getProduct(); //炒粉完成,交给客人品尝
    }

    3、这里是菜单A蛋炒面concreteBuilderA

    public ConcreteBuilderA implement Builder{
        Product p = new Product();
        public void addNoodles(){ //添加面
            p.add("noodles");
        }
        public void addEggOrMeat(){ //添加蛋
            p.add("egg");
        }
        public Product getProduct(){
            return p;
        }
    }

    这里是菜单B肉炒面concreteBuilderB

    public ConcreteBuilderB implement Builder{
        Product p = new Product();
        public void addNoodles(){//添加面条
            p.add("noodles");
        }
        public void addEggOrMeat(){//添加
            p.add("meat");
        }
        public Product getProduct(){
            return p;
        }
    }

    4、指挥者厨师director,按照菜单的要求,不能少放一颗食盐,也不能少放一滴食油,更不能少放面条,或者少放鸡蛋或肉。

    public Director{
    
        public void buildProject(Builder b){
            b.addNoodles();
            b.addEggOrMeat();
        }
    }

    最后是有客人来点餐(客户端)

        public static void main(String[] args) {
            Director d = new Director(); //厨师上班
            Builder b1 = new concreteBuilderA(); //拿到菜单A蛋炒面
            Builder b2 = new concreteBuilderB(); //拿到菜单B肉炒面
            d.buildProduct(b1); //有客人点了蛋炒面,厨师拿着菜单A去做
            d.buildProduct(b2); //有客人点了肉炒面,厨师拿着菜单B去做
            Product p1 = b1.getProduct(); //菜单A蛋炒面完成,将蛋炒面放到前台
            Product p2 = b2.getProduct(); //菜单B肉炒面完成,将肉炒面放到前台
        }

    以上就是一个完整的建造者模式。下面我们来分析一下它的优点:

    1.封装性好。客人是不知道炒面是谁做的,也不知道做炒面的流程是什么,他只是点一碗蛋/肉炒面,厨师就会把做好的产品端出来。

    2.扩展性好。在我们上面的例子中只支持蛋炒面和肉炒面,假如有客人需要蛋炒饭,厨师只需要再拿到蛋炒饭的菜单C,按照菜单上面的步骤,将addNoodles()改为添加addRice()即可。客人并不知道我们的饭店发生了什么,他甚至什么都不需要修改,一切都是由我们厨师在后台负责处理好了。同时也不会影响之前的菜单A和菜单B的口味,他们依然各司其职。

    3.产品更规范。我们的产品是固定了的炒面,它的步骤已经在厨师director中固定了,里面包含了添加食盐几克,添加食油几毫升,之后不论是什么时候,什么客人,只要来到这家饭店,吃到的炒面中添加的食盐和食油都是固定的。即不会出现时而太咸,时而太淡的情况。同时也避免由于步骤繁琐而遗漏其中一步或几步的情况。

    与工厂模式区别:

    建造者模式与工厂模式类似,他们都是建造者模式,适用的场景也很相似。一般来说,如果产品的建造很复杂,那么请用工厂模式;如果产品的建造更复杂,那么请用建造者模式。

  • 相关阅读:
    查看SQL Server被锁的表以及如何解锁【转】
    JQUERY的$(function(){})和window.onload=function(){}的区别【转】
    安装和使用Redis【转】
    RabbitMQ的简单应用【转】
    Redis集群的搭建【转】
    Spring--如何解决循环依赖
    分布式事务--2PC(两阶段提交)
    CAP理论
    JVM垃圾回收机制
    Redis面试题
  • 原文地址:https://www.cnblogs.com/yxth/p/7485573.html
Copyright © 2011-2022 走看看