成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java 8 重構傳統設計模式,是真的優雅

開發 前端
什么時候該用lambda,什么時候不用,這是要看情況的,如果處理邏輯相對比較簡單,可以用lamdba來重構,以便讓代碼更簡潔易讀,如果處理邏輯很復雜,應該還是用“類”。?

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. 模板方法

類圖如下,核心思路是把一些通用的標準方法,在抽象父類里僅定義方法簽名,實現邏輯交給子類。

比如:會員系統中,每個商家都會有一些營銷活動,需要推送某種信息給會員,但是不同的商家推送的內容可能不同,有些需要推送優惠券,有些需要積分通知。

抽象模板類:

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<Object[]> execute) {
        System.out.println("準備推送...");
        Object[] param = new Object[]{customerId, shopName};
        execute.accept(param);
        System.out.println("推送完成\n");
    }
}

借助Consumer<T>這個function interface,可以省去實現子類,具體的實現留到使用時再來決定,如:

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<Observer> list = new ArrayList<>();
    @Override
    public void registerObserver(Observer o) {
        list.add(o);
    }
    @Override
    public void notifyAllObserver(String orderNo) {
        list.forEach(c -> c.notify(orderNo));
    }
}

觀察者的二個實現:

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<Observer> list = new ArrayList<>();

    default void registerObserver(Observer o) {
        list.add(o);
    }

    default void nofityAllObserver(String orderNo) {
        list.forEach(c -> c.notify(orderNo));
    }
}

使用:

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 abstractclass 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<String> process(String param);
}

同樣的效果,可以寫得很簡潔:

static void test2() {
    Consumer<String> p1 = param -> System.out.println("processor 1 is processing:" + param);
    Consumer<String> p2 = param -> System.out.println("processor 2 is processing:" + param);
    p2.andThen(p1).accept("something happened");
}

andThen天然就是getNextProcessor的另一種表達。

重要提示:什么時候該用lambda,什么時候不用,這是要看情況的,如果處理邏輯相對比較簡單,可以用lamdba來重構,以便讓代碼更簡潔易讀,如果處理邏輯很復雜,應該還是用“類”。

責任編輯:武曉燕 來源: 一安未來
相關推薦

2022-08-19 08:39:34

OrderMySql數據庫

2024-08-06 09:43:54

Java 8工具編程

2021-11-16 12:02:29

Java代碼集合

2023-11-02 09:02:55

Java模式

2012-05-15 01:16:19

開發重構Java

2024-10-14 11:08:53

程序異常延遲

2015-08-11 09:39:25

重構提高代碼質量

2024-02-23 08:57:42

Python設計模式編程語言

2017-03-20 18:03:51

2021-11-29 10:27:24

設計模式程序員

2013-08-15 10:58:47

云計算SaaS企業應用

2019-01-15 10:49:57

傳統存儲私有云塊存儲

2013-05-22 10:30:57

SDN軟件定義網絡網絡架構

2019-03-21 15:30:05

JavaStream性能

2022-07-04 07:37:51

模板模式重構

2021-07-29 14:42:55

設計模式Java

2020-05-14 14:48:15

架構模式單庫

2010-03-04 09:46:38

Ubuntu PC

2018-10-07 06:30:40

代碼設計模式面向對象原則

2020-12-24 06:00:27

Python編程語言開發
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人不卡一区二区 | 色综合天天综合网国产成人网 | 日韩欧美亚洲一区 | 给我免费的视频在线观看 | 天天干成人网 | 午夜视频一区二区三区 | 亚洲精品久久久久久一区二区 | 亚洲女人的天堂 | 一级黄色毛片免费 | 国产视频一区二区 | 美女视频一区二区三区 | 极品一区 | 精品中文字幕一区 | 国产真实精品久久二三区 | 美女黄色在线观看 | 韩国av网站在线观看 | 日本福利在线观看 | 国产免费一区二区 | 日韩高清国产一区在线 | 亚洲精品在线免费 | 天堂网中文字幕在线观看 | 成人av影院 | 久久亚洲一区 | 亚洲国产精品一区二区久久 | 91精品久久久| 国产精品无码专区在线观看 | 国产综合久久 | 91免费视频 | 欧美精品久久一区 | 久热爱 | 欧美天堂 | 欧美日韩一区二区三区四区 | 狠狠狠干| 国产精品久久久久久久久久免费看 | 国产高潮好爽受不了了夜夜做 | 一区二区三区电影网 | 呦呦在线视频 | 久久99深爱久久99精品 | 中文字幕在线精品 | 国产在线观看一区二区三区 | 日韩成人免费视频 |