Themeplate Method
?
public abstract class Application {
protected abstract void init();
protected abstract void idle();
protected abstract void cleanup();
private boolean isDone = false;
protected void setDone(){
this.isDone = true;
}
protected boolean done(){
return isDone;
}
public void run(){
init();
while(!done())
idle();
cleanup();
}
}
?
public class WorkTemplateMethod extends Application {
public static void main(String[] args){
new WorkTemplateMethod().run();
}
@Override
protected void cleanup() {
System.out.println("clean up.");
}
@Override
protected void idle() {
System.out.println("idle.");setDone();
}
@Override
protected void init() {
System.out.println("init.");
}
}
?
Template Method模式展示了面向對象編程中諸多經典重用形式的一種。其中通用算法run()被放置在基類中,并且通過繼承在不同的具體上下文中實現該通用算法。
但這項技術也是有代價的。繼承是一種非常強的關系,派生類不可避免地要和它們的基類綁定在一起。
如果有個類Application2也需要WorkTemplateMethod中的idle()方法。然而卻沒法重用,由于繼承了Application,就注定把WorkTemplateMethod永遠地和Application綁定在了一起。
這時,我們就需要Strategy模式。
Strategy模式
?
public class ApplicationRunner {
private Application app = null;
public ApplicationRunner(Application app){
this.app = app;
}
public void run(){
app.init();
while(!app.done())
app.idle();
app.cleanup();
}
}
?
?
public interface Application {
public void init();
public void idle();
public void cleanup();
public boolean done();
}
?
?
?
public class WorkStrategy implements Application {
private boolean isDone = false;
@Override
public void cleanup() {
System.out.println("clean up.");
}
@Override
public boolean done() {
return isDone;
}
@Override
public void idle() {
System.out.println("idle.");
isDone = true;
}
@Override
public void init() {
System.out.println("init.");
}
public static void main(String[] args) {
new ApplicationRunner(new WorkStrategy()).run();
}
}
?
WorkStrategy對ApplicationRunner一無所知(main方法作為一個調用的例子,通常它都會在測試類中),它不依賴于run邏輯的任何實現方式。這和TemplateMethod模式是不同的,WorkTemplateMethod完全依賴于它的父類Application中run的邏輯,因而違反了DIP原則,而Strategy方法中不包含這中依賴。因此,當有別的邏輯出現時,也可以復用WorkStrategy實例中的方法。
?
?
public class ApplicationRunner2 {
private Application app = null;
public ApplicationRunner2(Application app){
this.app = app;
}
public void go(){
//app.init(); 不調用init方法
while(!app.done())
app.idle();
app.cleanup();
}
}
?
?這樣,只要使用new ApplicationRunner2(new WorkStrategy()).go()就可以了。
因此,Strategy模式比TemplateMethod模式多推薦了一個額外的好處。盡管TemplateMethod模式允許一個通用算法(run邏輯)操作多個可能的具體實現,但是由于Strategy模式完全遵循DIP原則,從而請允許每個具體實現都可以被多個不兩只的通用算法(run邏輯或go邏輯)操縱。
一句話,少用繼承,多用接口。
?
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061
微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元

