偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

趣談設(shè)計(jì)模式,你學(xué)會(huì)了嗎?

開(kāi)發(fā) 前端
雖然Java設(shè)計(jì)模式有23種,但是工作中常用的可能并沒(méi)有那麼多。就像新華字典有多少字,你沒(méi)必要都學(xué)一樣。本章我們只談常用的幾種設(shè)計(jì)模式,通過(guò)設(shè)計(jì)模式的理念、規(guī)約、到應(yīng)用,理解實(shí)戰(zhàn)中如何正確使用設(shè)計(jì)模式,不論對(duì)面試還是實(shí)際工作中都有益處。

背景

談起設(shè)計(jì)模式,你一定會(huì)問(wèn)?這玩意到底有啥用?我好像沒(méi)用過(guò)也不影響做一個(gè)碼農(nóng)。也可能項(xiàng)目中實(shí)際用了,但是你不自知。雖然Java設(shè)計(jì)模式有23種,但是工作中常用的可能并沒(méi)有那麼多。就像新華字典有多少字,你沒(méi)必要都學(xué)一樣。本章我們只談常用的幾種設(shè)計(jì)模式,通過(guò)設(shè)計(jì)模式的理念、規(guī)約、到應(yīng)用,理解實(shí)戰(zhàn)中如何正確使用設(shè)計(jì)模式,不論對(duì)面試還是實(shí)際工作中都有益處。

文章提綱

圖片圖片

設(shè)計(jì)理念

最為Java開(kāi)發(fā)者,程序員基本修養(yǎng)名言絕句:

  • 該露露,該藏藏
  • 該封裝的要封裝
  • 萬(wàn)事萬(wàn)物兼對(duì)象
  • 程序代碼要健壯

我們簡(jiǎn)單歸納為2個(gè)核心詞:高內(nèi)聚、低耦合。

很小的時(shí)候看過(guò)動(dòng)畫(huà)片,封神演義中哪吒:三頭八臂顯威力,千征百戰(zhàn)斗魔法。(串臺(tái)了。。。)

圖片圖片

我們根據(jù)這首歌詞抽象一下,哪吒:三頭八臂是靜態(tài)特征,千征百戰(zhàn)是動(dòng)態(tài)技能。把這些特征歸納映射一下:類(lèi) = 屬性 + 方法如下圖:

圖片圖片

以上是高內(nèi)聚的概念,什么是低耦合?

如果可能,我寫(xiě)一本神話(huà)《封神演戲》,說(shuō)哪吒有:三頭九臂。你肯定和我吵吵,要給它再配一把兵器。雖然還沒(méi)想好是啥,但是有個(gè)總則:絕對(duì)不影響先前這八臂的演技。這就是低耦合?。。∷^程序健壯、拓展性強(qiáng),也是這個(gè)道理。我們真誠(chéng)地希望:

圖片圖片

上述例子不是特別恰當(dāng),但是對(duì)于設(shè)計(jì)模式,我們終級(jí)的理念是:封裝變化的內(nèi)容,保留不變的宗旨。

設(shè)計(jì)原則

設(shè)計(jì)原則可以歸納為2大類(lèi):

  • 開(kāi)閉原則(李氏替換,組合復(fù)用,依賴(lài)倒置)

規(guī)定:軟件中的對(duì)象(類(lèi)、模塊、函數(shù)等等)應(yīng)該對(duì)于擴(kuò)展是開(kāi)放的,但是對(duì)于修改是封閉的。換句話(huà)說(shuō),一個(gè)實(shí)體是允許在不改變它的源代碼的前提下變更它的行為。

  • 單一職責(zé)(接口隔離,迪米特法則)

規(guī)定:一個(gè)類(lèi)只應(yīng)該有一個(gè)職責(zé),只有一個(gè)改變它的原因

Spring中的設(shè)計(jì)模式

在Spring框架中,各種設(shè)計(jì)模式被廣泛應(yīng)用以支持其強(qiáng)大的功能和靈活性。下面我將結(jié)合Spring的源碼,鑒賞下Spring中常見(jiàn)的幾種設(shè)計(jì)模式。

1. 單例模式

Spring框架中的Bean默認(rèn)就是單例的。Spring IoC容器負(fù)責(zé)創(chuàng)建對(duì)象實(shí)例,并確保在整個(gè)應(yīng)用中,針對(duì)同一個(gè)Bean的ID,只實(shí)例化一次對(duì)象。DefaultSingletonBeanRegistry類(lèi)是Spring管理單例Bean的核心類(lèi)。

// DefaultSingletonBeanRegistry類(lèi)中的部分源碼
public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {
    // ...
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
    // ...
    @Override
    public Object getSingleton(String beanName) {
        return getSingleton(beanName, true);
    }
    
    protected Object getSingleton(String beanName, boolean allowEarlyReference) {
        // ... 省略部分代碼
        Object singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
            synchronized (this.singletonObjects) {
                singletonObject = this.earlySingletonObjects.get(beanName);
                if (singletonObject == null && allowEarlyReference) {
                    ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                    if (singletonFactory != null) {
                        singletonObject = singletonFactory.getObject();
                        this.earlySingletonObjects.put(beanName, singletonObject);
                        this.singletonFactories.remove(beanName);
                    }
                }
            }
        }
        return (singletonObject != NULL_OBJECT ? singletonObject : null);
    }
    // ...
}

2. 工廠(chǎng)模式

Spring使用工廠(chǎng)模式通過(guò)BeanFactory、ApplicationContext等接口創(chuàng)建和管理Bean對(duì)象。DefaultListableBeanFactory是Spring中Bean工廠(chǎng)的實(shí)現(xiàn)類(lèi)。

// DefaultListableBeanFactory類(lèi)中的部分源碼
public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory
        implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {
    // ...
    @Override
    public <T> T getBean(String name, Class<T> requiredType) throws BeansException {
        return doGetBean(name, requiredType, null, false);
    }

    @Override
    public <T> T getBean(Class<T> requiredType) throws BeansException {
        return doGetBean(null, requiredType, null, false);
    }
    
    // ... 省略部分代碼
}

3. 代理模式

Spring AOP(面向切面編程)的實(shí)現(xiàn)就是基于代理模式。Spring創(chuàng)建目標(biāo)對(duì)象的代理對(duì)象,并在代理對(duì)象中織入切面邏輯。JdkDynamicAopProxy和CglibAopProxy是Spring AOP中創(chuàng)建代理的兩個(gè)核心類(lèi)。

// JdkDynamicAopProxy類(lèi)中的部分源碼
public class JdkDynamicAopProxy implements AopProxy, InvocationHandler, Serializable {
    // ...
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // ... 省略部分代碼
        // 獲取AdvisedSupport對(duì)象,包含了切面等AOP相關(guān)信息
        final AdvisedSupport advised = this.advised;
        // ... 省略部分代碼
        // 獲取攔截器鏈(切面鏈)
        List<Object> chain = advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);
        // ... 省略部分代碼
        // 執(zhí)行鏈?zhǔn)秸{(diào)用
        return invokeJoinpointUsingReflection(target, method, args, targetClass, chain);
    }
    // ...
}

4. 觀(guān)察者模式(監(jiān)聽(tīng)模式)

在Spring中,事件處理機(jī)制就是基于觀(guān)察者模式實(shí)現(xiàn)的。當(dāng)事件發(fā)生時(shí),所有注冊(cè)的觀(guān)察者都會(huì)收到通知并作出響應(yīng)。ApplicationEventMulticaster接口和SimpleApplicationEventMulticaster類(lèi)是Spring事件處理機(jī)制的核心。

// SimpleApplicationEventMulticaster類(lèi)中的部分源碼
public class SimpleApplicationEventMulticaster extends AbstractApplicationEventMulticaster {
    // ...
    @Override
    public void multicastEvent(final ApplicationEvent event, @Nullable ResolvableType eventType) {
        ResolvableType type = (eventType != null ? eventType : resolveDefaultEventType(event));
        for (ApplicationListener<?> listener : getApplicationListeners(event, type)) {```java
            // 調(diào)用監(jiān)聽(tīng)器的方法處理事件
            invokeListener(listener, event);
        }
    }

    private void invokeListener(ApplicationListener<?> listener, ApplicationEvent event) {
        ErrorHandler errorHandler = getErrorHandler();
        if (errorHandler != null) {
            try {
                doInvokeListener(listener, event);
            } catch (Throwable err) {
                errorHandler.handleError(err);
            }
        } else {
            doInvokeListener(listener, event);
        }
    }

    private void doInvokeListener(ApplicationListener<?> listener, ApplicationEvent event) {
        try {
            // 調(diào)用監(jiān)聽(tīng)器的onApplicationEvent方法
            listener.onApplicationEvent(event);
        } catch (ClassCastException ex) {
            // ... 省略部分代碼,處理類(lèi)型不匹配異常
        }
    }
    // ...
}

5. 責(zé)任鏈模式

在Spring中,HandlerInterceptor和HandlerInterceptorAdapter等類(lèi)在處理請(qǐng)求攔截時(shí),采用的就是責(zé)任鏈模式。一個(gè)請(qǐng)求會(huì)按照定義的攔截器順序,逐個(gè)被處理,直到找到對(duì)應(yīng)的處理器或者遍歷完所有的攔截器。

// HandlerInterceptor接口定義
public interface HandlerInterceptor {
    boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception;

    void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
            ModelAndView modelAndView) throws Exception;

    void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
            throws Exception;
}

// 實(shí)現(xiàn)HandlerInterceptor接口的自定義攔截器
public class CustomInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception {
        // 在這里執(zhí)行前置處理邏輯
        return true; // 返回true表示繼續(xù)向下執(zhí)行,返回false表示中斷請(qǐng)求
    }

    // ... 其他方法實(shí)現(xiàn)
}

6. 模版模式

Spring中的JdbcTemplate、HibernateTemplate等類(lèi)就是模版模式的典型應(yīng)用。它們定義了一個(gè)操作數(shù)據(jù)庫(kù)或Hibernate的骨架方法,允許子類(lèi)在不改變算法結(jié)構(gòu)的情況下重定義某些步驟的具體內(nèi)容。

// JdbcTemplate部分源碼
public class JdbcTemplate extends JdbcAccessor implements JdbcOperations, BeanFactoryAware {
    // ...
    public <T> T query(String sql, RowMapper<T> rowMapper) {
        return query(sql, new Object[0], rowMapper);
    }

    public <T> T query(String sql, Object[] args, RowMapper<T> rowMapper) {
        return query(sql, args, rowMapper, true);
    }

    // ... 省略部分代碼,這里是模版方法的實(shí)現(xiàn)
    // 真正的SQL執(zhí)行和結(jié)果集處理邏輯在這里,但是允許子類(lèi)通過(guò)RowMapper來(lái)定制結(jié)果集的處理方式
    // ...
}

// 自定義RowMapper實(shí)現(xiàn)
public class CustomRowMapper implements RowMapper<MyObject> {
    @Override
    public MyObject mapRow(ResultSet rs, int rowNum) throws SQLException {
        // 在這里定制如何從ResultSet中映射到MyObject對(duì)象
        return new MyObject(/* 映射邏輯 */);
    }
}

這些設(shè)計(jì)模式在Spring框架中被廣泛應(yīng)用,能夠靈活地應(yīng)對(duì)各種復(fù)雜場(chǎng)景,提供強(qiáng)大且可擴(kuò)展的功能。

實(shí)戰(zhàn)應(yīng)用

假設(shè)有這樣一個(gè)需求:

  1. 業(yè)務(wù)登錄商城用戶(hù)鑒權(quán)
  2. 購(gòu)買(mǎi)產(chǎn)品下訂單
  3. 校驗(yàn)訂單填寫(xiě)是否合法
  4. 記錄接口中的參數(shù)
  5. 訂單確認(rèn)后給買(mǎi)家發(fā)短信通知

根據(jù)業(yè)務(wù)場(chǎng)景,我們大致可拆分為:用戶(hù)流程、訂單流程

用戶(hù)登錄流程

在用戶(hù)登錄流程中,可能用到攔截器做鑒權(quán)校驗(yàn),日志記錄接口參數(shù)等,使用了一些常見(jiàn)的設(shè)計(jì)模式。

場(chǎng)景一、用戶(hù)鑒權(quán)校驗(yàn)

責(zé)任鏈模式

攔截器通常按照定義的順序執(zhí)行,每個(gè)攔截器檢查特定的條件或執(zhí)行特定的任務(wù)。

// 攔截器接口
public interface Interceptor {
    boolean intercept(AuthenticationContext context);
}

// 用戶(hù)校驗(yàn)攔截器
public class UserValidationInterceptor implements Interceptor {
    @Override
    public boolean intercept(AuthenticationContext context) {
        // 用戶(hù)校驗(yàn)邏輯
        if (isValidUser(context.getUser())) {
            return true;
        }
        return false;
    }

    private boolean isValidUser(User user) {
        // 校驗(yàn)用戶(hù)是否有效
        return true; // 示例,實(shí)際中應(yīng)有具體校驗(yàn)邏輯
    }
}

// 鑒權(quán)校驗(yàn)攔截器
public class AuthorizationInterceptor implements Interceptor {
    @Override
    public boolean intercept(AuthenticationContext context) {
        // 鑒權(quán)校驗(yàn)邏輯
        if (isAuthorized(context.getUser(), context.getCredentials())) {
            return true;
        }
        return false;
    }

    private boolean isAuthorized(User user, Credentials credentials) {
        // 校驗(yàn)用戶(hù)是否有權(quán)限
        return true; // 示例,實(shí)際中應(yīng)有具體校驗(yàn)邏輯
    }
}

// 攔截器鏈
public class InterceptorChain {
    private List<Interceptor> interceptors = new ArrayList<>();

    public void addInterceptor(Interceptor interceptor) {
        interceptors.add(interceptor);
    }

    public boolean execute(AuthenticationContext context) {
        for (Interceptor interceptor : interceptors) {
            if (!interceptor.intercept(context)) {
                // 如果攔截器返回false,則中斷鏈的執(zhí)行
                return false;
            }
        }
        return true;
    }
}

場(chǎng)景二、記錄用戶(hù)登錄信息

單例模式

日志記錄器通常設(shè)計(jì)為單例,確保全局只有一個(gè)實(shí)例。

Logger logger = LoggerFactory.getLoggerFactoryInstance().getLogger();  
logger.log("This is a user log message");

在這行代碼中,體現(xiàn)了單例模式的核心思想,確保了無(wú)論多少次調(diào)用LoggerFactory.getLoggerFactoryInstance(),都只會(huì)返回一個(gè)LoggerFactory實(shí)例。事實(shí)上,我們從源碼中也可以看到。

訂單流程

根據(jù)業(yè)務(wù)場(chǎng)景,核心訂單流程如下:

場(chǎng)景三、訂單校驗(yàn)

工廠(chǎng)模式

使用工廠(chǎng)模式實(shí)現(xiàn)的CheckOrderFactory,它用于創(chuàng)建不同類(lèi)型的訂單校驗(yàn)服務(wù)實(shí)例。

同時(shí),我們定義一個(gè)校驗(yàn)接口ICheckOrderService,并創(chuàng)建了兩個(gè)實(shí)現(xiàn)類(lèi):購(gòu)買(mǎi)數(shù)量校驗(yàn):CountCheckOrder和訂單參數(shù)校驗(yàn):ParamCheckOrder。

public interface ICheckOrderService {
    boolean checkOrder(Object order);
    String getErrorMessage();
}

購(gòu)買(mǎi)數(shù)量校驗(yàn)的實(shí)現(xiàn)類(lèi)CountCheckOrder:

public class CountCheckOrder implements ICheckOrderService {
    @Override
    public boolean checkOrder(Object order) {
        // 假設(shè)order是一個(gè)包含購(gòu)買(mǎi)數(shù)量的對(duì)象
        int quantity = ((Order) order).getQuantity();
        return quantity > 0; // 只允許購(gòu)買(mǎi)數(shù)量大于0
    }

    @Override
    public String getErrorMessage() {
        return "購(gòu)買(mǎi)數(shù)量必須大于0。";
    }
}

訂單參數(shù)校驗(yàn)的實(shí)現(xiàn)類(lèi)ParamCheckOrder:

public class ParamCheckOrder implements ICheckOrderService {
    @Override
    public boolean checkOrder(Object order) {
        // 假設(shè)order是一個(gè)包含各種訂單參數(shù)的對(duì)象
        // 這里可以添加具體的訂單參數(shù)校驗(yàn)邏輯
        return true; // 示例代碼,默認(rèn)返回true
    }

    @Override
    public String getErrorMessage() {
        return "訂單參數(shù)校驗(yàn)失敗。";
    }
}

工廠(chǎng)類(lèi)CheckOrderFactory,用于創(chuàng)建不同類(lèi)型的校驗(yàn)服務(wù)實(shí)例:

public class CheckOrderFactory {
    public static ICheckOrderService createCheckOrderService(String type) {
        switch (type) {
            case "count":
                return new CountCheckOrder();
            case "param":
                return new ParamCheckOrder();
            default:
                throw new IllegalArgumentException("不支持的校驗(yàn)類(lèi)型: " + type);
        }
    }
}

使用工廠(chǎng)模式有利于業(yè)務(wù)類(lèi)的實(shí)現(xiàn)和拓展,但是有時(shí)候也存在過(guò)度設(shè)計(jì),導(dǎo)致寫(xiě)了很多的業(yè)務(wù)類(lèi)。

場(chǎng)景四、短信通知

觀(guān)察者模式(監(jiān)聽(tīng)模式)

在Spring框架中,我們可以使用ApplicationEvent和ApplicationListener來(lái)實(shí)現(xiàn)事件發(fā)布和監(jiān)聽(tīng)的功能。

假設(shè)我們要下發(fā)一個(gè)購(gòu)買(mǎi)成功的短信提醒,那么就可以發(fā)布一個(gè)自定義的PurchaseSuccessEvent事件。

import org.springframework.context.ApplicationEvent;

public class PurchaseSuccessEvent extends ApplicationEvent {
    private final String buyerPhoneNumber;
    private final String orderId;

    public PurchaseSuccessEvent(Object source, String buyerPhoneNumber, String orderId) {
        super(source);
        this.buyerPhoneNumber = buyerPhoneNumber;
        this.orderId = orderId;
    }

    public String getBuyerPhoneNumber() {
        return buyerPhoneNumber;
    }

    public String getOrderId() {
        return orderId;
    }
}

接著定義一個(gè)SmsNotificationListener類(lèi),它實(shí)現(xiàn)了ApplicationListener接口,用于監(jiān)聽(tīng)PurchaseSuccessEvent事件:

import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;

@Component
public class SmsNotificationListener implements ApplicationListener<PurchaseSuccessEvent> {
    
    @Override
    public void onApplicationEvent(PurchaseSuccessEvent event) {
        String message = "親愛(ài)的買(mǎi)家,您的訂單 " + event.getOrderId() + " 購(gòu)買(mǎi)成功!";
        sendSms(event.getBuyerPhoneNumber(), message);
    }

    private void sendSms(String phoneNumber, String message) {
        // 在這里實(shí)現(xiàn)發(fā)送短信的邏輯
        System.out.println("Sending SMS to " + phoneNumber + ": " + message);
    }
}

然后,在Spring的配置中啟用事件發(fā)布功能。配置一個(gè)ApplicationEventPublisher的bean:

import org.springframework.context.ApplicationEventPublisher;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.support.GenericApplicationContext;

@Configuration
public class AppConfig {
    
    @Bean
    public ApplicationEventPublisher applicationEventPublisher() {
        return new GenericApplicationContext();
    }
}

最后,在業(yè)務(wù)邏輯中發(fā)布PurchaseSuccessEvent事件。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.stereotype.Service;

@Service
public class PurchaseService {

    private final ApplicationEventPublisher applicationEventPublisher;

    @Autowired
    public PurchaseService(ApplicationEventPublisher applicationEventPublisher) {
        this.applicationEventPublisher = applicationEventPublisher;
    }

    public void completePurchase(String buyerPhoneNumber, String orderId) {
        // 模擬購(gòu)買(mǎi)完成的業(yè)務(wù)邏輯
        // ...

        // 發(fā)布購(gòu)買(mǎi)成功事件
        applicationEventPublisher.publishEvent(new PurchaseSuccessEvent(this, buyerPhoneNumber, orderId));
    }
}

總結(jié)

  • 使用設(shè)計(jì)模式的宗旨:封裝變化的部分,維護(hù)不變的宗旨
  • 好處:提高代碼拓展性,程序更優(yōu)雅更健壯
  • 對(duì)開(kāi)源框架設(shè)計(jì)模式的使用,要提高鑒賞能力
  • 考慮可讀性,不可為了設(shè)計(jì)模式而過(guò)度設(shè)計(jì)
責(zé)任編輯:武曉燕 來(lái)源: 碼易有道
相關(guān)推薦

2024-05-09 08:14:09

系統(tǒng)設(shè)計(jì)語(yǔ)言多語(yǔ)言

2024-03-05 10:09:16

restfulHTTPAPI

2023-08-01 12:51:18

WebGPT機(jī)器學(xué)習(xí)模型

2024-01-02 12:05:26

Java并發(fā)編程

2024-01-19 08:25:38

死鎖Java通信

2023-07-26 13:11:21

ChatGPT平臺(tái)工具

2023-01-10 08:43:15

定義DDD架構(gòu)

2024-02-04 00:00:00

Effect數(shù)據(jù)組件

2022-12-06 07:53:33

MySQL索引B+樹(shù)

2023-01-31 08:02:18

2022-07-13 08:16:49

RocketMQRPC日志

2023-03-26 22:31:29

2023-08-26 21:34:28

Spring源碼自定義

2023-07-30 22:29:51

BDDMockitoAssert測(cè)試

2024-02-02 11:03:11

React數(shù)據(jù)Ref

2023-10-06 14:49:21

SentinelHystrixtimeout

2023-06-26 13:08:52

GraphQL服務(wù)數(shù)據(jù)

2023-05-05 06:54:07

MySQL數(shù)據(jù)查詢(xún)

2022-06-16 07:50:35

數(shù)據(jù)結(jié)構(gòu)鏈表

2023-10-26 07:15:46

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)