一.Builder模式

二.使用例子

三.Spring中的Builder模式

 

Builder模式,构建者、构造者模式,在《图解设计模式》中归为 生成实例 一栏,该模式用于组装具有复杂结构的实例;

    当需要逐步获取对象的初始值时,可以使用Builder模式;

 

一.Builder模式角色

Builder:建造者,Builder角色负责定义用来生成实例的接口API,Builder角色中准备了用于生成实例的具体方法;

ConcreteBuilder:具体的建造者,负责实现Builder接口的的类,定义了生成实例时实际调用的方法,定义了获取最终生成结果的方法;

Director:监工,负责使用Builder角色的接口来生成实例,不依赖于ConcreteBuilder;

 

二.使用例子

Demo:比如有一段预约信息,将其解析为  预约 对象; 预约信息如下:”Date, November 5, Headcount, 250, City, Shanghai, DollarsPerHead, 60,HasSite, false”;

date为预约日期,headCount为预期来人数,city为城市,DollarsPerHead为人均预算,hasSite为是否提供地址(比如预约某场演出,是否给表演者提供地点了);

类结构图:

                                           

预约信息  Reservation.java

@Getter
@Setter
@ToString   //lombok的注解,方便代码查看以及编写
public class Reservation {
    //Date, November 5, Headcount, 250, City, Shanghai, DollarsPerHead, 60,HasSite, false
    private Date date;
    private int headCount;
    private String city;
    private double dollarsPerHead;
    private boolean hasSite;
}

 

ReservationBuilder.java

public interface ReservationBuilder {
    
    public ReservationBuilder futurize(Date date) ;
    
    public ReservationBuilder setCity(String city);
    
    public ReservationBuilder setDollarsPerHead(double dollarsPerHead);
    
    public ReservationBuilder setSite(boolean hasSite);
    
    public ReservationBuilder setHeadCount(int headCount);
    
    Reservation build();
}

 

 

DefaultReservationBuilder.java

public class DefaultReservationBuilder implements ReservationBuilder{
    private Date date;
    private int headCount;
    private String city;
    private double dollarsPerHead;
    private boolean hasSite;
    
    public ReservationBuilder futurize(Date date) {
        this.date=date;
        return this;
    }
    
    public ReservationBuilder setCity(String city) {
        this.city=city;
        return this;
    }
    
    public ReservationBuilder setDollarsPerHead(double dollarsPerHead) {
        this.dollarsPerHead=dollarsPerHead;
        return this;
    }
    
    public ReservationBuilder setSite(boolean hasSite) {
        this.hasSite=hasSite;
        return this;
    }
    
    public ReservationBuilder setHeadCount(int headCount) {
        this.headCount=headCount;
        return this;
    }
    
    public Reservation build() {
        Reservation reservation = new Reservation();
        reservation.setCity(this.city);
        reservation.setDate(this.date);
        reservation.setDollarsPerHead(this.dollarsPerHead);
        reservation.setHasSite(this.hasSite);
        reservation.setHeadCount(this.headCount);
        return reservation;
    }
}

 

 

ReservationDirector.java

public class ReservationDirector {

    private ReservationBuilder builder;
    public ReservationDirector(ReservationBuilder builder) {
        this.builder=builder;
    }
    
    public Reservation construct(String input) throws Exception {
        String[] strings = input.split(",\\s*");
        for(int i=0;i<strings.length-1;i++) {
            String type=strings[i];
            String val=strings[i+1];
            if("date".equalsIgnoreCase(type)) {
                int year = Calendar.getInstance().get(Calendar.YEAR);
                String res=year+ " "+val.substring(0, 3)+" "+val.substring(val.length()-2);
                SimpleDateFormat sdf = new SimpleDateFormat("yyyy MMM dd", Locale.ENGLISH);
                Date date = sdf.parse(res);
                builder.futurize(date);
            }else if("headcount".equalsIgnoreCase(type)) {
                builder.setHeadCount(Integer.valueOf(val));
            }else if("city".equalsIgnoreCase(type)) {
                builder.setCity(val);
            }else if("dollarsperHead".equalsIgnoreCase(type)) {
                builder.setDollarsPerHead(Double.parseDouble(val));
            }else if("hassite".equalsIgnoreCase(type)) {
                builder.setSite(Boolean.parseBoolean(val));
            }
        }
        Reservation reservation = builder.build();
        return reservation;
    }
}

 

 

测试方法:

    public static void main(String[] args) throws Exception {
        //Date, November 5, Headcount, 250, City, Shanghai, DollarsPerHead, 60,HasSite, false
        String input="Date, November 5, Headcount, 20, City, Shanghai, DollarsPerHead, 60,HasSite, false";
        ReservationBuilder builder=new DefaultReservationBuilder();
        ReservationDirector director=new ReservationDirector(builder);
        Reservation reservation = director.construct(input);
        System.out.println(reservation);
    }

 

 

测试输出:

 

三.Spring中的Builder模式

  Spring中 RequestMappingInfo中内部接口Builder 、内部静态类DefaultBuilder等,就是采用了Builer模式,RequestMappingInfo就是监工Director对象;类结构图:

              

 

    public interface Builder {
        /* Set the path patterns*/
        Builder paths(String... paths);

        /* Set the request method conditions*/
        Builder methods(RequestMethod... methods);
/* Set the request param conditions*/ Builder params(String... params); /*Set the header conditions*/ Builder headers(String... headers); /* Set the consumes conditions*/ Builder consumes(String... consumes); /* Set the produces conditions*/ Builder produces(String... produces); /* Set the mapping name*/ Builder mappingName(String name); /* Set a custom condition to use*/ Builder customCondition(RequestCondition<?> condition); /* Provide additional configuration needed for request mapping purposes*/ Builder options(BuilderConfiguration options); /** Build the RequestMappingInfo*/ RequestMappingInfo build(); }

 

创建RequestMappingInfo对象地方:作用就是解析RequestMapping注解生成对应RequestMappingInfo对象;    

Builder方法大部分返回值为Builder类型,作用是为了链式执行,看起来生成RequestMappingInfo对象代码只有一句简洁;

 

版权声明:本文为lvbinbin2yujie原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/lvbinbin2yujie/p/10350847.html