Android设计模式

创建型设计模式——Builder模式

2017-12-21  本文已影响33人  忘尘无憾

定义

将一个复杂对象的构建与它的表示进行分离,使得同样的构建过程可以创建不同的表示。

对象创建型的设计模式

Builder模式是一步步创建一个复杂对象的创建模式,它允许用户在不知道内部构建细节的情况下,可以更精细的控制对象的构造流程。使得构造过程和部件都可以自由扩展,两者间的耦合也降到最低。

关键点

Builder模式UML类图

使用场景

场景例子说明

假设要组装一台电脑,简化为构建主机、设置操作系统、设置显示器三部分。

public class Computer {
    private String mBoard;//主板
    private String mDisplay;//显示器
    private String mOS;//操作系统
}

我们可以在构造方法中进行定义,用什么样的主板、显示器、操作系统,如下:
构造器模式

public Computer(String mBoard, String mDisplay, String mOS) {
    this.mBoard = mBoard;
    this.mDisplay = mDisplay;
    this.mOS = mOS;
}

但是当我们属性越来越多的时候,构造方法的方式显得不实用了。代码可读性变差,难以维护,调用者调用复杂等一系列缺点。

于是我们可以用Setter、Getter方法(JavaBean模式)。然而这种方式也存在或多或少的问题,比如当有大量的属性存在后,可能在前面已经set了主板,却再后面又重新修改了主板,这样很容易造成不安全的情况。

//属性越多,调用者重复越多
computer.setBoard("");
computer.setDisplay("");
computer.setOS("");
.....

//修改已经设置好了的某个属性
computer.setBoard("修改后的属性");

这时候可以考虑采用Builder模式。

经典的Builder模式
***
Product:被构造的复杂对象---->Computer 
***
//计算机抽象类,即Product角色
public abstract class Computer {
    protected String mBoard;//主机
    protected String mDisplay;//显示器
    protected String mOS;//操作系统

    private Computer() {
    }

    /**
     * 设置主机
     *
     * @return
     */
    public void setmBoard(String mBoard) {
        this.mBoard = mBoard;
    }

    /**
     * 设置显示器
     *
     * @param mDisplay
     */
    public void setmDisplay(String mDisplay) {
        this.mDisplay = mDisplay;
    }

    /**
     * 设置操作系统
     *
     * @param mOS
     */
    public void setmOS(String mOS) {
        this.mOS = mOS;
    }

    @Override
    public String toString() {
        return "Computer [mBorder=" + mBoard +
                ", mDisplay=" + mDisplay + ", mOs=" + mOS + "]";
    }
}
//具体的Computer类
public class MateBook extends Computer {
    protected MateBook() {
    }

    @Override
    public void setmOS() {
        mOS = "HUAWEI MateBook Android 7.0";
    }
}
***
Builder:抽象接口---->Builder  
Builder类负责的复杂对象的组成部分的装载 
***
//抽象Builder类
public abstract class Builder {
    //设置主板、显示器、操作系统
    public abstract void buildBoard(String board);
    public abstract void buildDisplay(String display);
    public abstract void buildOS();
    
    //创建Computer
    public abstract Computer create();
}
***
BuilderImpl:抽象接口的具体实现---->MateBookBuilder
***
//具体的Builder类
public class MateBookBuilder extends Builder {
    private Computer mComputer = new MateBook();

    @Override
    public void buildBoard(String board) {
        mComputer.setmBoard(board);
    }

    @Override
    public void buildDisplay(String display) {
        mComputer.setmDisplay(display);
    }

    @Override
    public void buildOS() {
        mComputer.setmOS();
    }

    @Override
    public Computer create() {
        return mComputer;
    }
}
***
Director:接口的构造者和使用者 ---->Goods
***
//Director类,复杂构造Computer
public class Director {
    Builder mBuilder = null;

    public Director(Builder builder) {
        mBuilder = builder;
    }

    /**
     * 构建对象
     *
     * @param board
     * @param display
     */
    public void construct(String board, String display) {
        mBuilder.buildBoard(board);
        mBuilder.buildDisplay(display);
        mBuilder.buildOS();
    }
}
public class ComputerTest {
    public static void main(String[] args) {
        //创建构建器
        Builder builder = new MateBookBuilder();
        //Director
        Director director = new Director(builder);
        //封装构建过程
        director.construct("XXX主板", "XXX显示器");
        //构建计算机,输出信息
        System.out.println(builder.create().toString());
    }
}
result:
Computer [mBorder=XXX主板, mDisplay=XXX显示器, mOs=HUAWEI MateBook Android 7.0]

上述例子中,通过具体的MateBookBuilder来构建MateBook对象,而Director封装了构建复杂产品对象的过程,对外隐藏构建细节。Builder与Director一起将一个复杂对象的构建与他的表示分离,使得同样的构建过程可以创建不同的对象。

Builder模式需要注意的地方

builder模式多处理是用来的聚合的观念,而不是继承的观念

这部分的理解非常重要

builder模式主要是用来创建复杂的对象,从而实现对对象的创建过程和表示层实现分离。
我们需要考虑下builder模式到底真正的含义是什么?
参考引文:
Android Builder模式
Builder模式的误区:将复杂对象的构建进行封装,就是Builder模式了吗?

变种Builder模式

变种Builder模式在现实Android开发中会经常用到。现实开发中,Director角色会经常被省略。而直接使用一个Builder来进行对象的组装,这个Builder通常为链式调用,它的关键点是每个setter方法都返回自身,即return this

调用如下:
new TestBuilder().setA("A").setB("B").create()

变种Builder模式和经典Builder模式区别

变种Builder的常用写法,Android中常见的链式调用

//变种Builder。Director角色会被省略
public class BMateBookBuilder {
    private Computer mComputer;

    public BMateBookBuilder() {
        mComputer = new MateBook();
    }

    public BMateBookBuilder(Computer computer) {
        mComputer = computer;
    }

    public BMateBookBuilder buildBoard(String board) {
        mComputer.setmBoard(board);
        return this;
    }

    public BMateBookBuilder buildDisplay(String display) {
        mComputer.setmDisplay(display);
        return this;
    }

    public BMateBookBuilder buildOS() {
        mComputer.setmOS();
        return this;
    }

    public Computer create() {
        return mComputer;
    }
}
//调用的时候如下:
new BMateBookBuilder()
    .buildBoard("TTT主板")
    .buildDisplay("TTT显示器")
    .buildOS()
    .create();

new BMateBookBuilder(new MateBook())
    .buildBoard("HHH主板")
    .buildDisplay("HHH显示器")
    .buildOS()
    .create();

//输出toString()
Computer [mBorder=TTT主板, mDisplay=TTT显示器, mOs=HUAWEI MateBook Android 7.0]
Computer [mBorder=HHH主板, mDisplay=HHH显示器, mOs=HUAWEI MateBook Android 7.0]

在Android中,Builder链式调用通常采用内部静态类构建Builder

public class BMateBook extends Computer {
    private BMateBook(Builder builder) {
        this.mBoard = builder.mBoard;
        this.mDisplay = builder.mDisplay;
        setmOS();
    }

    @Override
    public void setmOS() {
        mOS = "HUAWEI MateBook Android 7.0";
    }

    public static class Builder {
        private String mBoard;
        private String mDisplay;

        public Builder setBoard(String board) {
            this.mBoard = board;
            return this;
        }

        public Builder setDisplay(String display) {
            this.mDisplay = display;
            return this;
        }

        public BMateBook build() {
            return new BMateBook(this);
        }
    }
}
//调用方式:
new BMateBook.Builder()
    .setBoard("NNN主板")
    .setDisplay("NNN显示器")
    .build()
//输出toString()
Computer [mBorder=NNN主板, mDisplay=NNN显示器, mOs=HUAWEI MateBook Android 7.0]

如何保证Builder模式下线程安全?

public class AQ_BMateBook {
    //都设置为final类型,不可变对象,以保证线程的安全
    private final String board;
    private final String display;
    private final String os;

    //传入builder对象,通过builder对象来初始化参数
    private AQ_BMateBook(Builder builder) {
        this.board = builder.mBoard;
        this.display = builder.mDisplay;
        this.os = "HUAWEI MateBook Android 7.0";
    }

    @Override
    public String toString() {
        return "Computer [mBorder=" + board +
                ", mDisplay=" + display + ", mOs=" + os + "]";
    }

    public static class Builder {
        private String mBoard;
        private String mDisplay;
        private String mOS;

        public Builder setBoard(String board) {
            this.mBoard = board;
            return this;
        }

        public Builder setDisplay(String display) {
            this.mDisplay = display;
            return this;
        }

        public Builder setOS(String os) {
            this.mOS = os;
            return this;
        }

        public AQ_BMateBook create() {
            return new AQ_BMateBook(this);
        }
    }
}

对于线程安全这部分了解的不够明白,如有错误望指正。

Builder模式的优缺点

优点:

缺点:



本文引用了很多内容,自身对Builder模式的理解还不够深刻,若有不正确的地方,望指正。

引用:
《Android源码设计模式解析与实战》
Android中建造者(builder)模式
Android Builder模式
Builder模式的误区:将复杂对象的构建进行封装,就是Builder模式了吗?
Android中的构建者(Builder)模式

上一篇 下一篇

猜你喜欢

热点阅读