Java 8 重構傳統設計模式,是真的優雅!
來源:Java後端面試官
java8中提供的很多新特性可以用來重構傳統設計模式中的寫法,下面是一些示例:
1. 策略模式
上圖是策略模式的類圖,假設我們現在要儲存訂單,OrderService
介面定義要做什麼,而NoSqlSaveOrderStragegy
以及MySqlSaveOrderStrategy
則提供了二種策略,分別是儲存到nosql資料庫,以及傳統的mysql關係型資料庫,最後在OrderServiceExecutor
中透過建構函式注入最終要使用的策略。
傳統寫法,這個場景至少得4個類,程式碼如下:
OrderService
介面:
public interface OrderService {
void saveOrder(String orderNo);
}
Mysql策略實現:
public class MySqlSaveOrderStrategy implements OrderService {
@Override
public void saveOrder(String orderNo) {
System.out.println("order:" + orderNo + " save to mysql");
}
}
Nosql策略實現
public class NoSqlSaveOrderStrategy implements OrderService {
@Override
public void saveOrder(String orderNo) {
System.out.println("order:" + orderNo + " save to nosql");
}
}
使用策略的輔助"容器"
public class OrderServiceExecutor {
private final OrderService service;
public OrderServiceExecutor(OrderService service) {
this.service = service;
}
public void save(String orderNo) {
this.service.saveOrder(orderNo);
}
}
執行測試類:
public class OrderServiceTest {
public static void main(String[] args) {
OrderServiceExecutor executor1 = new OrderServiceExecutor(new MySqlSaveOrderStrategy());
executor1.save("001");
OrderServiceExecutor executor2 = new OrderServiceExecutor(new NoSqlSaveOrderStrategy());
executor2.save("002");
}
}
重構後,可以省去2個策略實現類,程式碼如下:
public static void main(String[] args) {
OrderServiceExecutor executor1 = new OrderServiceExecutor((String orderNo) -> System.out.println("order:" + orderNo + " save to mysql"));
executor1.save("001");
OrderServiceExecutor executor2 = new OrderServiceExecutor((String orderNo) -> System.out.println("order:" + orderNo + " save to nosql"));
executor2.save("002");
}
2. 模板方法
類圖如下,核心思路是把一些通用的標準方法,在抽象父類裡僅定義方法簽名,實現邏輯交給子類。
比如:會員系統中,每個商家都會有一些營銷活動,需要推送某種資訊給會員,但是不同的商家推送的內容可能不同,有些需要推送優惠券,有些需要積分通知。關注公眾號:Java後端面試官,回覆關鍵詞:“2022Java面經” ,獲取大廠面試真題
抽象模板類:
public abstract class AbstractPushTemplate {
public void push(int customerId, String shopName) {
System.out.println("準備推送...");
execute(customerId, shopName);
System.out.println("推送完成\n");
}
abstract protected void execute(int customerId, String shopName);
}
優惠券的具體模板
public class PushCouponTemplate extends AbstractPushTemplate {
@Override
protected void execute(int customerId, String shopName) {
System.out.println("會員:" + customerId + ",你好," + shopName + "送您一張優惠券");
}
}
積分的具體模板
public class PushScoreTemplate extends AbstractPushTemplate {
@Override
protected void execute(int customerId, String shopName) {
System.out.println("會員:" + customerId + ",你好," + shopName + "送您10個積分");
}
}
使用示例:
AbstractPushTemplate template1 = new PushCouponTemplate();
template1.push(1, "糖果店");
AbstractPushTemplate template2 = new PushScoreTemplate();
template2.push(1, "服裝店");
顯然如果模板的實現方式越多,子類就越多。
使用java8重構後,可以把上面的3個模板(包括抽象類别範本)減少到1個,參考下面:
public class PushTemplateLambda {
public void push(int customerId, String shopName, Consumer {
System.out.println("準備推送...");
Object[] param = new Object[]{customerId, shopName};
execute.accept(param);
System.out.println("推送完成\n");
}
}
藉助Consumer
,可以省去實現子類,具體的實現留到使用時再來決定,如:
new PushTemplateLambda().push(1, "糖果店", (Object[] obj) -> {
System.out.println("會員:" + obj[0] + ",你好," + obj[1] + "送您一張優惠券");
});
new PushTemplateLambda().push(1, "服裝店", (Object[] obj) -> {
System.out.println("會員:" + obj[0] + ",你好," + obj[1] + "送您10個積分");
});
3. 觀察者模式
思路:基於某個Subject主題,然後一堆觀察者Observer註冊到主題上,有事件發生時,subject根據註冊列表,去通知所有的observer。
Observer介面:
public interface Observer {
void notify(String orderNo);
}
Subject介面:
public interface Subject {
void registerObserver(Observer o);
void notifyAllObserver(String orderNo);
}
Subject介面實現:
public class SubjectImpl implements Subject {
private final List
觀察者的二個實現:
OrderObserver:
public class OrderObserver implements Observer {
@Override
public void notify(String orderNo) {
System.out.println("訂單 " + orderNo + " 狀態更新為【已支付】");
}
}
StockObserver:
public class StockObserver implements Observer {
@Override
public void notify(String orderNo) {
System.out.println("訂單 " + orderNo + " 已通知庫房發貨!");
}
}
測試一把:
static void test1() {
Subject subject = new SubjectImpl();
subject.registerObserver(new OrderObserver());
subject.registerObserver(new StockObserver());
subject.notifyAllObserver("001");
}
用java8重構後,介面可以提供預設實現方法,我們弄一個新的主題介面
public interface NewSubject {
List
使用:
static void test2() {
NewSubject subject = new NewSubject() {
};
subject.registerObserver((String orderNo) -> System.out.println("訂單 " + orderNo + " 狀態更新為【已支付】"));
subject.registerObserver((String orderNo) -> System.out.println("訂單 " + orderNo + " 已通知庫房發貨!"));
subject.nofityAllObserver("002");
}
只用2個介面實現了觀察者模式。
4. 責任鏈/職責鏈模式
核心思想:每個處理環節,都有一個“指標”指向下一個處理者,類似連結串列一樣。
Processor介面:
public interface Processor {
Processor getNextProcessor();
void process(String param);
}
抽象實現類
public abstract class AbstractProcessor implements Processor {
private Processor next;
public AbstractProcessor(Processor processor) {
this.next = processor;
}
@Override
public Processor getNextProcessor() {
return next;
}
@Override
public abstract void process(String param);
}
定義2個具體的實現
public class ProcessorImpl1 extends AbstractProcessor {
public ProcessorImpl1(Processor processor) {
super(processor);
}
@Override
public void process(String param) {
System.out.println("processor 1 is processing:" + param);
if (getNextProcessor() != null) {
getNextProcessor().process(param);
}
}
}
及
public class ProcessorImpl2 extends AbstractProcessor {
public ProcessorImpl2(Processor next) {
super(next);
}
@Override
public void process(String param) {
System.out.println("processor 2 is processing:" + param);
if (getNextProcessor() != null) {
getNextProcessor().process(param);
}
}
}
使用示例:
static void test1() {
Processor p1 = new ProcessorImpl1(null);
Processor p2 = new ProcessorImpl2(p1);
p2.process("something happened");
}
用java8重構後,只需要一個新介面
@FunctionalInterface
public interface NewProcessor {
Consumer;
}
同樣的效果,可以寫得很簡潔:
static void test2() {
Consumer
andThen天然就是getNextProcessor的另一種表達。
重要提示:什麼時候該用lambda,什麼時候不用,這是要看情況的,如果處理邏輯相對比較簡單,可以用lamdba來重構,以便讓程式碼更簡潔易讀,如果處理邏輯很複雜,應該還是用“類”。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/70024922/viewspace-2952592/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 設計模式:如何優雅地使用工廠模式設計模式
- 如何設計優雅的類結構
- 設計模式:如何優雅地使用責任鏈模式設計模式
- 如何優雅的設計Java異常Java
- Java設計模式8:迭代器模式Java設計模式
- 【Java】設計模式--結構型模式Java設計模式
- RESTful & “優雅的”API 響應結構設計RESTAPI
- .NET領域驅動設計—看DDD是如何運用設計模式顛覆傳統架構設計模式架構
- 使用Lambdas重構工廠設計模式設計模式
- 重學 Java 設計模式:實戰建造者模式Java設計模式
- 重學 Java 設計模式:實戰原型模式Java設計模式原型
- (Java)設計模式:結構型Java設計模式
- 又被面試官問設計模式了,我真的是面試設計模式
- 【設計模式】你的單例模式真的是生產可用的嗎?設計模式單例
- 8.java設計模式之裝飾者模式Java設計模式
- 讓程式碼變得優雅簡潔的神器:Java8 Stream流式程式設計Java程式設計
- 重學 Java 設計模式:實戰抽象工廠模式Java設計模式抽象
- 重學 Java 設計模式:實戰工廠方法模式Java設計模式
- 重學 Java 設計模式:實戰介面卡模式Java設計模式
- Java用策略模式優雅地匯出ExcelJava模式Excel
- Laravel 實用小技巧 —— 如何優雅地設計方法傳參?Laravel
- 設計模式你真的懂了嗎?設計模式
- 怎樣用Java 8優雅的開發業務Java
- 設計模式系列·Adapter模式之如何優雅的使用別人的輪子設計模式APT
- 使用列舉來寫出更優雅的單例設計模式單例設計模式
- 如何優雅的在業務中使用設計模式(程式碼如詩)設計模式
- [譯文]優雅的現代JavaScript設計模式: 冰凍工廠JavaScript設計模式
- 如何優雅的設計 React 元件React元件
- 重學Java設計模式-學習筆記(1)Java設計模式筆記
- Java設計模式——模板設計模式Java設計模式
- Java 系統架構設計Java架構
- Java 8中處理集合的優雅姿勢——StreamJava
- java設計模式之單例模式你真的會了嗎?(懶漢式篇)Java設計模式單例
- 設計模式是在運用構造定律設計模式
- 使用函數語言程式設計重構模板模式函數程式設計模式
- 從重構的角度學習bridge設計模式設計模式
- Java軟體架構設計慨論(轉載)--設計模式和系統架構的關係Java架構設計模式
- 重學設計模式-單例模式設計模式單例