欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java設(shè)計(jì)模式中的適配器模式詳解

 更新時(shí)間:2023年05月17日 09:29:15   作者:知了一笑  
適配器模式(Adapter),是Java23種設(shè)計(jì)模式中的結(jié)構(gòu)型模式之一,它可以將新的功能和原先的功能連接起來(lái),使由于需求變動(dòng)導(dǎo)致不能用的功能,重新利用起來(lái),本文將詳細(xì)聊一聊Java適配器的運(yùn)用場(chǎng)景和使用方法,感興趣的同學(xué)可以跟著小編一起來(lái)學(xué)習(xí)

一、概述

適配器模式(Adapter),是23種設(shè)計(jì)模式中的結(jié)構(gòu)型模式之一;它就像我們電腦上接口不夠時(shí),需要用到的拓展塢,起到轉(zhuǎn)接的作用。它可以將新的功能和原先的功能連接起來(lái),使由于需求變動(dòng)導(dǎo)致不能用的功能,重新利用起來(lái)。

上圖的Mac上,只有兩個(gè)typec接口,當(dāng)我們需要用到USB、網(wǎng)線、HDMI等接口時(shí),這就不夠用了,所以我們需要一個(gè)拓展塢來(lái)增加電腦的接口

言歸正傳,下面來(lái)了解下適配器模式中的角色:請(qǐng)求者(client)、目標(biāo)角色(Target)、源角色(Adaptee)、適配器角色(Adapter),這四個(gè)角色是保證這個(gè)設(shè)計(jì)模式運(yùn)行的關(guān)鍵。

  • client:需要使用適配器的對(duì)象,不需要關(guān)心適配器內(nèi)部的實(shí)現(xiàn),只對(duì)接目標(biāo)角色。
  • Target:目標(biāo)角色,和client直接對(duì)接,定義了client需要用到的功能。
  • Adaptee:需要被進(jìn)行適配的對(duì)象。
  • Adapter:適配器,負(fù)責(zé)將源對(duì)象轉(zhuǎn)化,給client做適配。

二、入門案例

適配器模式也分兩種:對(duì)象適配器、類適配器。其實(shí)兩種方式的區(qū)別在于,適配器類中的實(shí)現(xiàn),類適配器是通過(guò)繼承源對(duì)象的類,對(duì)象適配器是引用源對(duì)象的類。

當(dāng)然兩種方式各有優(yōu)缺點(diǎn),咱分別來(lái)說(shuō)下;

類適配器:由于采用繼承模式,在適配器中可以重寫Adaptee原有的方法,使得適配器可以更加靈活;但是有局限性,Java是單繼承模式,所以適配器類只能繼承Adaptee,不能在額外繼承其他類,也導(dǎo)致Target類只能是接口。

對(duì)象適配器:這個(gè)模式規(guī)避了單繼承的劣勢(shì),將Adaptee類用引用的方式傳遞給Adapter,這樣可以傳遞的是Adaptee對(duì)象本身及其子類對(duì)象,相比類適配器更加的開(kāi)放;但是也正是因?yàn)檫@種開(kāi)放性,導(dǎo)致需要自己重新定義Adaptee,增加額外的操作。

類適配器UML圖

對(duì)象適配器UML圖

下面,是結(jié)合上面電腦的場(chǎng)景,寫的一個(gè)入門案例,分別是四個(gè)類:Client、Adaptee、Adapter、Target,代表了適配器模式中的四種角色。

/**
 * @author
 * @version 1.0
 * @date 2023/5/9 15:54
 * @description:源角色
 */
public class Adaptee {
    /**
     * 需要被適配的適配的功能
     * 以Mac筆記本的typec接口舉例
     */
    public void typeC() {
        System.out.println("我只是一個(gè)typeC接口");
    }
}
/**
 * @author 
 * @version 1.0
 * @date 2023/5/9 15:57
 * @description:目標(biāo)接口
 */
public interface Target {

    /**
     * 定義一個(gè)轉(zhuǎn)接功能的入口
     */
    void socket();
}
/**
 * @author 
 * @version 1.0
 * @date 2023/5/9 16:00
 * @description:適配器
 */
public class Adapter extends Adaptee implements Target {

    /**
     * 實(shí)現(xiàn)適配功能
     * 以Mac的拓展塢為例,拓展更多的接口:usb、typc、網(wǎng)線插口...
     */
    @Override
    public void socket() {
        typeC();
        System.out.println("新增usb插口。。。");
        System.out.println("新增網(wǎng)線插口。。。");
        System.out.println("新增typec插口。。。");
    }
}
/**
 * @author 
 * @version 1.0
 * @date 2023/5/9 15:52
 * @description:請(qǐng)求者
 */
public class Client {

    public static void main(String[] args) {
        Target target = new Adapter();
        target.socket();
    }
}

這個(gè)案例比較簡(jiǎn)單,僅僅是一個(gè)入門的demo,也是類適配器模式的案例,采用繼承模式。在對(duì)象適配器模式中,區(qū)別就是Adapter這個(gè)適配器類,采用的是組合模式,下面是對(duì)象適配器模式中Adapter的代碼;

/**
 * @author 
 * @version 1.0
 * @date 2023/5/9 16:00
 * @description:適配器
 */
public class Adapter implements Target {

    private Adaptee adaptee;

    public Adapter(Adaptee adaptee) {
        this.adaptee = adaptee;
    }

    /**
     * 實(shí)現(xiàn)適配功能
     * 以Mac的拓展塢為例,拓展更多的接口:usb、typc、網(wǎng)線插口...
     */
    @Override
    public void socket() {
        adaptee.typeC();
        System.out.println("新增usb插口。。。");
        System.out.println("新增網(wǎng)線插口。。。");
        System.out.println("新增typec插口。。。");
    }
}

三、運(yùn)用場(chǎng)景

其實(shí)適配器模式為何會(huì)存在,全靠“爛代碼”的襯托。在初期的設(shè)計(jì)上,一代目沒(méi)有考慮到后期的兼容性問(wèn)題,只顧自己一時(shí)爽,那后期接手的人就會(huì)感覺(jué)到頭疼,就會(huì)有“還不如重寫這段代碼的想法”。但是這部分代碼往往都是經(jīng)過(guò)N代人的充分測(cè)試,穩(wěn)定性比較高,一時(shí)半會(huì)還不能對(duì)它下手。這時(shí)候我們的適配器模式就孕育而生,可以在不動(dòng)用老代碼的前提下,實(shí)現(xiàn)新邏輯,并且能做二次封裝。這種場(chǎng)景,我在之前的系統(tǒng)重構(gòu)中深有體會(huì),不說(shuō)了,都是淚。

當(dāng)然還存在一種情況,可以對(duì)不同的外部數(shù)據(jù)進(jìn)行統(tǒng)一輸出。例如,寫一個(gè)獲取一些信息的接口,你對(duì)前端暴露的都是統(tǒng)一的返回字段,但是需要調(diào)用不同的外部api獲取不同的信息,不同的api返回給你的字段都是不同的,比如企業(yè)工商信息、用戶賬戶信息、用戶津貼信息等等。下面我對(duì)這種場(chǎng)景具體分析下;

首先,我定義一個(gè)接口,接收用戶id和數(shù)據(jù)類型兩個(gè)參數(shù),定義統(tǒng)一的輸出字段。

/**
 * @author
 * @version 1.0
 * @date 2023/5/10 11:03
 * @description
 */
@RestController
@RequestMapping("/user")
@RequiredArgsConstructor
public class UserInfoController {

    private final UserInfoTargetService userInfoTargetService;

    @PostMapping("/info")
    public Result<DataInfoVo> queryInfo(@RequestParam Integer userId, @RequestParam String type) {
        return Result.success(userInfoTargetService.queryData(userId, type));
    }
}

定義統(tǒng)一的輸出的類DataInfoVo,這里定義的字段需要暴露給前端,具體業(yè)務(wù)意義跟前端商定。

/**
 * @author
 * @version 1.0
 * @date 2023/5/10 14:40
 * @description
 */
@Data
public class DataInfoVo {
    /**
     * 名稱
     */
    private String name;
    /**
     * 類型
     */
    private String type;
    /**
     * 預(yù)留字段:具體業(yè)務(wù)意義自行定義
     */
    private Object extInfo;
}

然后,定義Target接口(篇幅原因,這里不做展示),Adapter適配器類,這里采用的是對(duì)象適配器,由于單繼承的限制,對(duì)象適配器也是最常用的適配器模式。

/**
 * @author 
 * @version 1.0
 * @date 2023/5/10 15:09
 * @description
 */
@Service
@RequiredArgsConstructor
public class UserInfoAdapter implements UserInfoTargetService {
    /**
     * 源數(shù)據(jù)類管理器
     */
    private final AdapteeManager adapteeManager;

    @Override
    public DataInfoVo queryData(Integer userId, String type) {
        // 根據(jù)類型,得到唯一的源數(shù)據(jù)類
        UserBaseAdaptee adaptee = adapteeManager.getAdaptee(type);
        if (Objects.nonNull(adaptee)) {
            Object data = adaptee.getData(userId, type);
            return adaptee.convert(data);
        }
        return null;
    }
}

這里定義了一個(gè)AdapteeManager類,表示管理Adaptee類,內(nèi)部維護(hù)一個(gè)map,用于存儲(chǔ)真實(shí)Adaptee類。

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/10 15:37
 * @description
 */
public class AdapteeManager {

    private Map<String, UserBaseAdaptee> baseAdapteeMap;

    public void setBaseAdapteeMap(List<UserBaseAdaptee> adaptees) {
        baseAdapteeMap = adaptees.stream()
                .collect(Collectors.toMap(handler -> AnnotationUtils.findAnnotation(handler.getClass(), Adapter.class).type(), v -> v, (v1, v2) -> v1));
    }

    public UserBaseAdaptee getAdaptee(String type) {
        return baseAdapteeMap.get(type);
    }
}

最后,按照數(shù)據(jù)類型,定義了三個(gè)Adaptee類:AllowanceServiceAdaptee(津貼)、BusinessServiceAdaptee(企業(yè)工商)、UserAccountServiceAdaptee(用戶賬戶)。

/**
 * @author
 * @version 1.0
 * @date 2023/5/10 15:00
 * @description
 */
@Adapter(type = "JT")
public class AllowanceServiceAdaptee implements UserBaseAdaptee {

    @Override
    public Object getData(Integer userId, String type) {
        // 模擬調(diào)用外部api,查詢津貼信息
        AllowanceVo allowanceVo = new AllowanceVo();
        allowanceVo.setAllowanceType("管理津貼");
        allowanceVo.setAllowanceAccount("xwqeretry2345676");
        allowanceVo.setAmount(new BigDecimal(20000));
        return allowanceVo;
    }

    @Override
    public DataInfoVo convert(Object data) {
        AllowanceVo preConvert = (AllowanceVo) data;
        DataInfoVo dataInfoVo = new DataInfoVo();
        dataInfoVo.setName(preConvert.getAllowanceAccount());
        dataInfoVo.setType(preConvert.getAllowanceType());
        dataInfoVo.setExtInfo(preConvert.getAmount());
        return dataInfoVo;
    }
}
/**
 * @author 
 * @version 1.0
 * @date 2023/5/10 15:00
 * @description
 */
@Adapter(type = "QY")
public class BusinessServiceAdaptee implements UserBaseAdaptee {

    @Override
    public Object getData(Integer userId, String type) {
        // 模擬調(diào)用外部api,查詢企業(yè)工商信息
        BusinessVo businessVo = new BusinessVo();
        businessVo.setBusName("xxx科技有限公司");
        businessVo.setBusCode("q24243Je54sdfd99");
        businessVo.setBusType("中大型企業(yè)");
        return businessVo;
    }

    @Override
    public DataInfoVo convert(Object data) {
        BusinessVo preConvert = (BusinessVo) data;
        DataInfoVo dataInfoVo = new DataInfoVo();
        dataInfoVo.setName(preConvert.getBusName());
        dataInfoVo.setType(preConvert.getBusType());
        dataInfoVo.setExtInfo(preConvert.getBusCode());
        return dataInfoVo;
    }
}
/**
 * @author 
 * @version 1.0
 * @date 2023/5/10 15:00
 * @description
 */
@Adapter(type = "YH")
public class UserAccountServiceAdaptee implements UserBaseAdaptee {

    @Override
    public Object getData(Integer userId, String type) {
        // 模擬調(diào)用外部api,查詢企業(yè)工商信息
        UserAccountVo userAccountVo = new UserAccountVo();
        userAccountVo.setAccountNo("afsdfd1243567");
        userAccountVo.setAccountType("銀行卡");
        userAccountVo.setName("中國(guó)農(nóng)業(yè)銀行");
        return userAccountVo;
    }

    @Override
    public DataInfoVo convert(Object data) {
        UserAccountVo preConvert = (UserAccountVo) data;
        DataInfoVo dataInfoVo = new DataInfoVo();
        dataInfoVo.setName(preConvert.getName());
        dataInfoVo.setType(preConvert.getAccountType());
        dataInfoVo.setExtInfo(preConvert.getAccountNo());
        return dataInfoVo;
    }
}

這三個(gè)類都實(shí)現(xiàn)一個(gè)接口UserBaseAdaptee,該接口定義了統(tǒng)一的規(guī)范

/**
 * @author 
 * @version 1.0
 * @date 2023/5/10 15:03
 * @description
 */

public interface UserBaseAdaptee {
    /**
     * 獲取數(shù)據(jù)
     * @param userId
     * @param type
     * @return
     */
    Object getData(Integer userId, String type);

    /**
     * 數(shù)據(jù)轉(zhuǎn)化為統(tǒng)一的實(shí)體
     * @param data
     * @return
     */
    DataInfoVo convert(Object data);
}

這些類中,其實(shí)重點(diǎn)看下UserInfoAdapter適配器類,這里做的操作是通過(guò)源數(shù)據(jù)類,拿到外部返回的數(shù)據(jù),最后將不同的數(shù)據(jù)轉(zhuǎn)化為統(tǒng)一的字段,返回出去。

這里我沒(méi)有按照固定的模式,稍加了改變。將適配器類中引用源數(shù)據(jù)類的方式,改成將源數(shù)據(jù)類加入map中暫存,最后通過(guò)前端傳輸?shù)膖ype字段來(lái)獲取源數(shù)據(jù)類,這也是對(duì)象適配器比較靈活的一種體現(xiàn)。

四、源碼中的運(yùn)用

在JDK的源碼中,JUC下有個(gè)類FutureTask,其中它的一段構(gòu)造方法如下:

public class FutureTask<V> implements RunnableFuture<V> {
    public FutureTask(Callable<V> callable) {
        if (callable == null)
            throw new NullPointerException();
        this.callable = callable;
        this.state = NEW;       // ensure visibility of callable
    }
    
	public FutureTask(Runnable runnable, V result) {
        this.callable = Executors.callable(runnable, result);
        this.state = NEW;       // ensure visibility of callable
    }
}

其中一個(gè)構(gòu)造函數(shù)中,callable是通過(guò)Executors類的方法進(jìn)行適配的,通過(guò)一個(gè)RunnableAdapter的適配器類,進(jìn)行包裝并返回

public static <T> Callable<T> callable(Runnable task, T result) {
        if (task == null)
            throw new NullPointerException();
        return new RunnableAdapter<T>(task, result);
    }
static final class RunnableAdapter<T> implements Callable<T> {
        final Runnable task;
        final T result;
        RunnableAdapter(Runnable task, T result) {
            this.task = task;
            this.result = result;
        }
        public T call() {
            task.run();
            return result;
        }
    }

這樣的話,無(wú)論傳入Runnable還是Callable都可以適配任務(wù),雖然看著是調(diào)用了Callable的call方法,實(shí)際內(nèi)部是調(diào)用了Runnable的run方法,并且將傳入的返回?cái)?shù)據(jù)返回給外部使用。

五、總結(jié)

適配器模式其實(shí)是一個(gè)比較好理解的設(shè)計(jì)模式,但是對(duì)于大多數(shù)初學(xué)者而言,就會(huì)很容易看一遍之后立馬忘,這是缺少實(shí)際運(yùn)用造成的。其實(shí)編程主要考察的還是我們的一種思維模式,就像這個(gè)適配器模式,理解它的運(yùn)用場(chǎng)景最重要。如果給你一個(gè)業(yè)務(wù)場(chǎng)景,你能在腦海中有大致的設(shè)計(jì)思路或者解決方案,那你就已經(jīng)掌握精髓了。至于具體的落地,有些細(xì)節(jié)忘記也是在所難免,翻翻資料就會(huì)立馬回到腦海中。

最后,每次遇到問(wèn)題,用心總結(jié),你會(huì)離成功更近一步。

六、參考源碼

編程文檔:
https://gitee.com/cicadasmile/butte-java-note

應(yīng)用倉(cāng)庫(kù):
https://gitee.com/cicadasmile/butte-flyer-parent

以上就是Java設(shè)計(jì)模式中的適配器模式詳解的詳細(xì)內(nèi)容,更多關(guān)于Java 適配器模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • MyBatis攔截器實(shí)現(xiàn)分頁(yè)功能實(shí)例

    MyBatis攔截器實(shí)現(xiàn)分頁(yè)功能實(shí)例

    本篇文章主要介紹了MyBatis攔截器實(shí)現(xiàn)分頁(yè)功能實(shí)例,這里整理了詳細(xì)的代碼,有需要的小伙伴可以參考下。
    2017-04-04
  • Spring中@Controller和@RestController的區(qū)別詳解

    Spring中@Controller和@RestController的區(qū)別詳解

    這篇文章主要介紹了Spring中@Controller和@RestController的區(qū)別詳解,@RestController?是?@Controller?和?@ResponseBody?的結(jié)合體,單獨(dú)使用?@RestController?的效果與?@Controller?和?@ResponseBody?二者同時(shí)使用的效果相同,需要的朋友可以參考下
    2023-10-10
  • java直接插入排序示例

    java直接插入排序示例

    這篇文章主要介紹了java直接插入排序示例,插入排序的比較次數(shù)仍然是n的平方,但在一般情況下,它要比冒泡排序快一倍,比選擇排序還要快一點(diǎn)。它常常被用在復(fù)雜排序算法的最后階段,比如快速排序。
    2014-05-05
  • 學(xué)習(xí)Java之異常到底該如何捕獲和處理

    學(xué)習(xí)Java之異常到底該如何捕獲和處理

    我們知道,Java的異常處理是通過(guò)5個(gè)關(guān)鍵字來(lái)實(shí)現(xiàn)的,即try、catch、throw、throws和finally,try?catch語(yǔ)句用于捕獲并處理異常,但具體該怎么捕獲異常,怎么拋出異常,什么時(shí)候拋,什么時(shí)候捕,感興趣的小伙伴跟著小編一起來(lái)看看吧
    2023-08-08
  • 基于Java解決華為機(jī)試實(shí)現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換?

    基于Java解決華為機(jī)試實(shí)現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換?

    這篇文章主要介紹了基于Java解決華為機(jī)試實(shí)現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換,文章舉例說(shuō)明圍繞文章主題展開(kāi)相關(guān)內(nèi)容,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-02-02
  • Spring線程池ThreadPoolTaskExecutor的用法及說(shuō)明

    Spring線程池ThreadPoolTaskExecutor的用法及說(shuō)明

    這篇文章主要介紹了Spring線程池ThreadPoolTaskExecutor的用法及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-07-07
  • java裁剪圖片并保存的示例分享

    java裁剪圖片并保存的示例分享

    在這篇文章中我們將學(xué)習(xí)如何用Java 對(duì)圖像進(jìn)行剪裁并將剪裁出來(lái)的部分單獨(dú)保存到文件中
    2014-01-01
  • springboot2.x解決運(yùn)行順序及Bean對(duì)象注入順序的問(wèn)題

    springboot2.x解決運(yùn)行順序及Bean對(duì)象注入順序的問(wèn)題

    這篇文章主要介紹了springboot2.x解決運(yùn)行順序及Bean對(duì)象注入順序的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-01-01
  • Java面向?qū)ο蟪绦蛟O(shè)計(jì)多態(tài)性示例

    Java面向?qū)ο蟪绦蛟O(shè)計(jì)多態(tài)性示例

    這篇文章主要介紹了Java面向?qū)ο蟪绦蛟O(shè)計(jì)多態(tài)性,結(jié)合實(shí)例形式分析了java多態(tài)性的概念、原理、定義與使用方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下
    2018-03-03
  • redis深入淺出分布式鎖實(shí)現(xiàn)上篇

    redis深入淺出分布式鎖實(shí)現(xiàn)上篇

    在單體應(yīng)用中,如果我們對(duì)共享數(shù)據(jù)不進(jìn)行加鎖操作,會(huì)出現(xiàn)數(shù)據(jù)一致性問(wèn)題,我們的解決辦法通常是加鎖。下面我們一起聊聊使用redis來(lái)實(shí)現(xiàn)分布式鎖
    2022-08-08

最新評(píng)論