Android編程設(shè)計(jì)模式之Builder模式實(shí)例詳解
本文實(shí)例講述了Android編程設(shè)計(jì)模式之Builder模式。分享給大家供大家參考,具體如下:
一、介紹
Builder模式是一步一步創(chuàng)建一個(gè)復(fù)雜對(duì)象的創(chuàng)建型模式,它允許用戶(hù)在不知道內(nèi)部構(gòu)建細(xì)節(jié)的情況下,可以更精細(xì)的控制對(duì)象的構(gòu)造流程。該模式是為了將構(gòu)建復(fù)雜對(duì)象的過(guò)程和它的部件解耦,使得構(gòu)建過(guò)程和部件的表示隔離開(kāi)來(lái)。
因?yàn)橐粋€(gè)復(fù)雜的對(duì)象有很多大量組成部分,例如車(chē),有車(chē)輪、方向盤(pán)、發(fā)動(dòng)機(jī),還有各種小零件等,如何將這些部件裝配成一輛汽車(chē),這個(gè)裝配過(guò)程很漫長(zhǎng),也很復(fù)雜,對(duì)于這種情況,為了在構(gòu)建過(guò)程中對(duì)外部隱藏實(shí)現(xiàn)細(xì)節(jié),就可以使用Builder模式將部件和組裝過(guò)程分離,使得構(gòu)建過(guò)程和部件都可以自由擴(kuò)展,兩者之間的耦合也降到最低。
二、定義
將一個(gè)復(fù)雜對(duì)象的構(gòu)建與它的表示分離,使得同樣的構(gòu)建過(guò)程可以創(chuàng)建不同的表示。
三、使用場(chǎng)景
(1)相同的方法,不同的執(zhí)行順序,產(chǎn)生不同的事件結(jié)果時(shí)。
(2)多個(gè)部件或零件,都可以裝配到一個(gè)對(duì)象中,但是產(chǎn)生的運(yùn)行結(jié)果又不相同時(shí)。
(3)產(chǎn)品類(lèi)非常復(fù)雜,或者產(chǎn)品類(lèi)中的調(diào)用順序不同產(chǎn)生了不同的作用,這個(gè)時(shí)候使用建造者模式非常合適。
(4)當(dāng)初始化一個(gè)對(duì)象特別復(fù)雜,如參數(shù)多,且很多參數(shù)都具有默認(rèn)值時(shí)。
四、Builder模式的UML類(lèi)圖
角色介紹:
Product產(chǎn)品類(lèi)——產(chǎn)品的抽象類(lèi);
Builder——抽象Builder類(lèi),規(guī)范產(chǎn)品的組建,一般是由子類(lèi)實(shí)現(xiàn)具體的組建過(guò)程;
ConcreateBuilder——具體的Builder類(lèi);
Director——統(tǒng)一組裝過(guò)程;

五、Builder模式的簡(jiǎn)單實(shí)現(xiàn)
計(jì)算機(jī)的組裝過(guò)程較為復(fù)雜,并且組裝順序是不固定的,為了易于理解,我們把計(jì)算機(jī)的組裝過(guò)程簡(jiǎn)化為構(gòu)建主機(jī)、設(shè)置操作系統(tǒng)、設(shè)置顯示器3個(gè)部分,然后通過(guò)Director和具體的Builder來(lái)構(gòu)建計(jì)算機(jī)對(duì)象。
示例代碼:
/**
* 計(jì)算機(jī)抽象類(lèi),即Product角色
*/
public abstract class Computer {
protected String mBoard;
protected String mDisplay;
protected String mOS;
protected Computer(){}
/**
* 設(shè)置主板
* @param board
*/
public void setBoard(String board){
this.mBoard = board;
}
/**
* 設(shè)置顯示器
* @param display
*/
public void setDisplay(String display){
this.mDisplay = display;
}
/**
* 設(shè)置操作系統(tǒng)
*/
public abstract void setOS();
@Override
public String toString(){
return "Computer [mBoard=" + mBoard + ", mDisplay=" + mDisplay + ", mOS=" + mOS + "]";
}
}
/**
* 具體的Computer類(lèi),Macbook
*/
public class Macbook extends Computer {
protected Macbook(){}
@Override
public void setOS() {
mOS = "Mac OS X 10";
}
}
/**
* 抽象Builder類(lèi)
*/
public abstract class Builder {
/**
* 設(shè)置主機(jī)
* @param board
*/
public abstract void buildBoard(String board);
/**
* 設(shè)置顯示器
* @param display
*/
public abstract void buildDisplay(String display);
/**
* 設(shè)置操作系統(tǒng)
*/
public abstract void buildOS();
/**
* 創(chuàng)建Computer
* @return
*/
public abstract Computer create();
}
/**
* 具體的Builder類(lèi),MacbookBuilder
*/
public class MacbookBuilder extends Builder {
private Computer mComputer = new Macbook();
@Override
public void buildBoard(String board) {
mComputer.setBoard(board);
}
@Override
public void buildDisplay(String display) {
mComputer.setDisplay(display);
}
@Override
public void buildOS() {
mComputer.setOS();
}
@Override
public Computer create() {
return mComputer;
}
}
/**
* Director類(lèi),負(fù)責(zé)構(gòu)造Computer
*/
public class Director {
Builder mBuilder = null;
public Director(Builder builder){
mBuilder = builder;
}
/**
* 構(gòu)建對(duì)象
* @param board 主板
* @param display 顯示器
*/
public void construct(String board, String display){
mBuilder.buildBoard(board);
mBuilder.buildDisplay(display);
mBuilder.buildOS();
}
}
/**
* 測(cè)試代碼
*/
public class Test {
public static void main(String[] args){
//構(gòu)建器
Builder builder = new MacbookBuilder();
//Director
Director pcDirector = new Director(builder);
//封裝構(gòu)建過(guò)程
pcDirector.construct("英特爾主板","Retina顯示器");
//構(gòu)建計(jì)算機(jī),輸出相關(guān)信息
System.out.println("Computer Info : " + builder.create().toString());
}
}
輸出結(jié)果:
上述示例中,通過(guò)具體的MacbookBuilder來(lái)構(gòu)建Macbook對(duì)象,而Director封裝了構(gòu)建復(fù)雜產(chǎn)品對(duì)象的過(guò)程,對(duì)外隱藏構(gòu)建細(xì)節(jié)。Builder與Director一起將一個(gè)復(fù)雜的對(duì)象的構(gòu)建與它的表示分離,使得同樣的構(gòu)建過(guò)程可以創(chuàng)建不同的對(duì)象。
值得注意的是,在現(xiàn)實(shí)的開(kāi)發(fā)過(guò)程中,Director角色經(jīng)常會(huì)被省略。而直接使用一個(gè)Builder來(lái)進(jìn)行對(duì)象的組裝,這個(gè)Builder通常為鏈?zhǔn)秸{(diào)用,它的關(guān)鍵點(diǎn)是每個(gè)setter方法都返回自身,也就是return this,這樣就使得setter方法可以鏈?zhǔn)秸{(diào)用,代碼大致如下:
new TestBuilder()
.setA("A")
.create();
通過(guò)這種形式不僅去除了Director角色,整個(gè)結(jié)構(gòu)也更加簡(jiǎn)單,也能對(duì)Product對(duì)象的組裝過(guò)程有更精細(xì)的控制。
六、Builder模式變種——鏈?zhǔn)秸{(diào)用
示例代碼:
public class User {
private final String name; //必選
private final String cardID; //必選
private final int age; //可選
private final String address; //可選
private final String phone; //可選
private User(UserBuilder userBuilder){
this.name=userBuilder.name;
this.cardID=userBuilder.cardID;
this.age=userBuilder.age;
this.address=userBuilder.address;
this.phone=userBuilder.phone;
}
public String getName() {
return name;
}
public String getCardID() {
return cardID;
}
public int getAge() {
return age;
}
public String getAddress() {
return address;
}
public String getPhone() {
return phone;
}
public static class UserBuilder{
private final String name;
private final String cardID;
private int age;
private String address;
private String phone;
public UserBuilder(String name,String cardID){
this.name=name;
this.cardID=cardID;
}
public UserBuilder age(int age){
this.age=age;
return this;
}
public UserBuilder address(String address){
this.address=address;
return this;
}
public UserBuilder phone(String phone){
this.phone=phone;
return this;
}
public User build(){
return new User(this);
}
}
}
需要注意的點(diǎn):
User類(lèi)的構(gòu)造方法是私有的,調(diào)用者不能直接創(chuàng)建User對(duì)象。
User類(lèi)的屬性都是不可變的。所有的屬性都添加了final修飾符,并且在 構(gòu)造方法中設(shè)置了值。并且,對(duì)外只提供getters方法。
Builder的內(nèi)部類(lèi)構(gòu)造方法中只接收必傳的參數(shù),并且該必傳的參數(shù)使用了final修飾符。
調(diào)用方式:
new User.UserBuilder("Jack","10086")
.age(25)
.address("GuangZhou")
.phone("13800138000")
.build();
相比起前面通過(guò)構(gòu)造函數(shù)和setter/getter方法兩種方式,可讀性更強(qiáng)。唯一可能存在的問(wèn)題就是會(huì)產(chǎn)生多余的Builder對(duì)象,消耗內(nèi)存。然而大多數(shù)情況下我們的Builder內(nèi)部類(lèi)使用的是靜態(tài)修飾的(static),所以這個(gè)問(wèn)題也沒(méi)多大關(guān)系。
關(guān)于線程安全
Builder模式是非線程安全的,如果要在Builder內(nèi)部類(lèi)中檢查一個(gè)參數(shù)的合法性,必需要在對(duì)象創(chuàng)建完成之后再檢查
正確示例:
public User build() {
User user = new user(this);
if (user.getAge() > 120) {
throw new IllegalStateException("Age out of range"); // 線程安全
}
return user;
}
錯(cuò)誤示例:
public User build() {
if (age > 120) {
throw new IllegalStateException("Age out of range"); // 非線程安全
}
return new User(this);
}
七、用到Builder模式的例子
1、Android中的AlertDialog.Builder
private void showDialog(){
AlertDialog.Builder builder=new AlertDialog.Builder(context);
builder.setIcon(R.drawable.icon);
builder.setTitle("Title");
builder.setMessage("Message");
builder.setPositiveButton("Button1", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//TODO
}
});
builder.setNegativeButton("Button2", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//TODO
}
});
builder.create().show();
}
2、OkHttp中OkHttpClient的創(chuàng)建
OkHttpClient okHttpClient = new OkHttpClient.Builder()
.cache(getCache())
.addInterceptor(new HttpCacheInterceptor())
.addInterceptor(new LogInterceptor())
.addNetworkInterceptor(new HttpRequestInterceptor())
.build();
3、Retrofit中Retrofit對(duì)象的創(chuàng)建
Retrofit retrofit = new Retrofit.Builder()
.client(createOkHttp())
.addConverterFactory(GsonConverterFactory.create())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.baseUrl(BASE_URL)
.build();
可見(jiàn)在實(shí)際使用中,均省略掉了Director角色,在很多框架源碼中,涉及到Builder模式時(shí),大多都不是經(jīng)典GOF的Builder模式,而是選擇了結(jié)構(gòu)更加簡(jiǎn)單的后者。
八、優(yōu)缺點(diǎn)
優(yōu)點(diǎn):
良好的封裝性,使得客戶(hù)端不需要知道產(chǎn)品內(nèi)部實(shí)現(xiàn)的細(xì)節(jié)
建造者獨(dú)立,擴(kuò)展性強(qiáng)
缺點(diǎn):
產(chǎn)生多余的Builder對(duì)象、Director對(duì)象,消耗內(nèi)存
更多關(guān)于Android相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《Android開(kāi)發(fā)入門(mén)與進(jìn)階教程》、《Android調(diào)試技巧與常見(jiàn)問(wèn)題解決方法匯總》、《Android基本組件用法總結(jié)》、《Android視圖View技巧總結(jié)》、《Android布局layout技巧總結(jié)》及《Android控件用法總結(jié)》
希望本文所述對(duì)大家Android程序設(shè)計(jì)有所幫助。
相關(guān)文章
Android Studio導(dǎo)入jar包過(guò)程詳解
這篇文章主要介紹了Android Studio導(dǎo)入jar包過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11
Android 自定義view實(shí)現(xiàn)進(jìn)度條加載效果實(shí)例代碼
這篇文章主要介紹了Android 自定義view實(shí)現(xiàn)進(jìn)度條加載效果實(shí)例代碼,需要的朋友可以參考下2017-08-08
Android手機(jī)內(nèi)存中文件的讀寫(xiě)方法小結(jié)
這篇文章主要介紹了Android手機(jī)內(nèi)存中文件的讀寫(xiě)方法,實(shí)例總結(jié)了Android針對(duì)文件讀寫(xiě)操作的相關(guān)技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-04-04
Android中Paint類(lèi)和Canvas類(lèi)的方法匯總
本文主要介紹了Android中Paint類(lèi)和Canvas類(lèi)的方法。具有很好的參考價(jià)值,下面跟著小編一起來(lái)看下吧2017-02-02
Android利用Intent.ACTION_SEND進(jìn)行分享
這篇文章主要介紹了Android利用Intent.ACTION_SEND進(jìn)行分享,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
Android4.2中全屏或者取消標(biāo)題欄的方法總結(jié)
有的時(shí)候我們會(huì)看到,會(huì)先出現(xiàn)標(biāo)題欄,然后再消失,因?yàn)槲覀冎皇窃赼ctivity的oncreate方法中定義的,其他實(shí)現(xiàn)方法如下,感興趣的朋友可以了解下哈2013-06-06
Android 通過(guò)SQLite數(shù)據(jù)庫(kù)實(shí)現(xiàn)數(shù)據(jù)存儲(chǔ)管理
SQLiteOpenHelper 是Android 提供的一個(gè)抽象工具類(lèi),負(fù)責(zé)管理數(shù)據(jù)庫(kù)的創(chuàng)建、升級(jí)工作。本文主要介紹了如何使用SQLite數(shù)據(jù)庫(kù)實(shí)現(xiàn)對(duì)數(shù)據(jù)進(jìn)行存儲(chǔ)管理,感興趣的可以了解一下2021-11-11
APK包名修改 請(qǐng)問(wèn)如何修改APK包名
今天,想在android手機(jī)上安裝兩個(gè)相同的應(yīng)用,本以為可以安裝不同版本的,試了幾次,均相互覆蓋了,于是,只能設(shè)法修改apk所對(duì)應(yīng)的包名(package name),需要了解的朋友可以參考下2012-12-12
Android開(kāi)發(fā)中怎樣調(diào)用系統(tǒng)Email發(fā)送郵件(多種調(diào)用方式)
在Android中調(diào)用其他程序進(jìn)行相關(guān)處理,幾乎都是使用的Intent,所以,Email也不例外,所謂的調(diào)用Email,只是說(shuō)Email可以接收Intent并做這些事情2013-06-06

