創(chuàng)建型模式包括:工廠方法模式,抽象工廠方法模式,單例模式,建造者模式艇劫,原型模式
在這里我們選出前三種來進(jìn)行簡單的介紹
工廠方法模式:
簡單工廠方法模式:就是建立一個(gè)工廠類,對實(shí)現(xiàn)了同一接口的一些類進(jìn)行實(shí)例的創(chuàng)建惩激。首先看下關(guān)系圖:
#######工廠類的實(shí)現(xiàn)如下:
public class SendFactory {
public Sender produce(String type) {
if ("mail".equals(type)) {
return new MailSender();
} else if ("sms".equals(type)) {
return new SmsSender();
} else {
System.out.println("請輸入正確的類型!");
return null;
}
}
}
多個(gè)工廠方法模式:
#######工廠類的實(shí)現(xiàn):
public class SendFactory {
public Sender produceMail(){
return new MailSender();
}
public Sender produceSms(){
return new SmsSender();
}
}
靜態(tài)工廠方法模式
#######工廠類的實(shí)現(xiàn):
public class SendFactory {
public static Sender produceMail(){
return new MailSender();
}
public static Sender produceSms(){
return new SmsSender();
}
}
總體來說店煞,工廠模式適合:凡是出現(xiàn)了大量的產(chǎn)品需要?jiǎng)?chuàng)建,并且具有共同的接口時(shí)风钻,可以通過工廠方法模式進(jìn)行創(chuàng)建顷蟀。在以上的三種模式中,第一種如果傳入的字符串有誤骡技,不能正確創(chuàng)建對象鸣个,第三種相對于第二種,不需要實(shí)例化工廠類布朦,所以囤萤,大多數(shù)情況下,我們會(huì)選用第三種——靜態(tài)工廠方法模式是趴。
工廠方法模式的優(yōu)點(diǎn)
良好的封裝性涛舍,代碼結(jié)構(gòu)清晰。
工廠方法模式的擴(kuò)展性非常優(yōu)秀唆途。
屏蔽產(chǎn)品類富雅。
解耦。
抽象工廠模式
類圖:
工廠方法模式有一個(gè)問題就是窘哈,類的創(chuàng)建依賴工廠類吹榴,也就是說,如果想要拓展程序滚婉,必須對工廠類進(jìn)行修改,這違背了閉包原則帅刀,所以让腹,從設(shè)計(jì)角度考慮,有一定的問題扣溺,如何解決骇窍?就用到抽象工廠模式,創(chuàng)建多個(gè)工廠類锥余,這樣一旦需要增加新的功能腹纳,直接增加新的工廠類就可以了,不需要修改之前的代碼。因?yàn)槌橄蠊S不太好理解嘲恍,我們先看看圖足画,然后就和代碼,就比較容易理解佃牛。
例子:
public interface Sender {
public void Send();
}
//兩個(gè)實(shí)現(xiàn)類:
public class MailSender implements Sender {
@Override
public void Send() {
System.out.println("this is mailsender!");
}
}
public class SmsSender implements Sender {
@Override
public void Send() {
System.out.println("this is sms sender!");
}
}
//兩個(gè)工廠類:
public class SendMailFactory implements Provider {
@Override
public Sender produce(){
return new MailSender();
}
}
public class SendSmsFactory implements Provider{
@Override
public Sender produce() {
return new SmsSender();
}
}
//在提供一個(gè)接口:
public interface Provider {
public Sender produce();
}
其實(shí)這個(gè)模式的好處就是淹辞,如果你現(xiàn)在想增加一個(gè)功能:發(fā)及時(shí)信息,則只需做一個(gè)實(shí)現(xiàn)類俘侠,實(shí)現(xiàn)Sender接口象缀,同時(shí)做一個(gè)工廠類,實(shí)現(xiàn)Provider接口爷速,就OK了央星,無需去改動(dòng)現(xiàn)成的代碼。這樣做惫东,拓展性較好等曼!
單例模式:
http://www.reibang.com/p/d59c64480ed8
其中注意雙重檢查鎖定中使用volatile變量主要是禁止指令的重排序,在創(chuàng)建一個(gè)實(shí)例的時(shí)候主要有以下三步:
1.為對象分配內(nèi)存空間
2.初始化對象
3.將內(nèi)存空間的地址賦值給對應(yīng)引用
如果是這個(gè)流程凿蒜,多線程環(huán)境下就可能將一個(gè)未初始化的對象引用暴露出來禁谦,從而導(dǎo)致不可預(yù)料的結(jié)果(如題目的描述,這里就是因?yàn)?instance = new Singleton(); 不是原子操作废封,編譯器存在指令重排州泊,從而存在線程1 創(chuàng)建實(shí)例后(初始化未完成),線程2 判斷對象不為空后對其操作漂洋,但實(shí)際對象仍為空遥皂,造成錯(cuò)誤)。因此刽漂,為了防止這個(gè)過程的重排序演训,我們需要將變量設(shè)置為volatile類型的變量,volatile的禁止重排序保證了操作的有序性贝咙。
本文參考:http://blog.csdn.net/zhangerqing/article/details/8194653