結合JDK源碼看設計模式——適配器模式

来源:https://www.cnblogs.com/Cubemen/archive/2019/04/06/10661419.html
-Advertisement-
Play Games

定義: 將一個類的介面轉換成客戶期望的另外一個介面(重點理解適配的這兩個字),使得介面不相容的類可以一起工作適用場景: 詳解 首先來從生活中的常見場景來看,一個電源插座輸出都是220V,而我們一些電子設備,比如手機,MP3,MP4,所需要的電壓不一樣,也不可能直接就是220接上,這就需要一個中間的轉 ...


定義:
  將一個類的介面轉換成客戶期望的另外一個介面(重點理解適配的這兩個字),使得介面不相容的類可以一起工作
適用場景:

  1. 已經存在的類,它的方法和需求不匹配的時候
  2. 在軟體維護階段考慮的設計模式

詳解
  首先來從生活中的常見場景來看,一個電源插座輸出都是220V,而我們一些電子設備,比如手機,MP3,MP4,所需要的電壓不一樣,也不可能直接就是220接上,這就需要一個中間的轉換器,每個廠家不同,對應的充電線也有可能不同。這個不同的充電線就可以理解為一個適配器。而220V的輸出電壓可以看做是我們做好的一套系統,只不過對應到不同客戶就需要不同的適配器。下麵分為三個模塊來講解

1.類適配器

輸出的電壓類

public class AC220 {
public int outputAC220V(){
int output = 220;
System.out.println("輸出交流電"+output+"V");
return output;
}
}

5V電壓介面

public interface DC5 {
int outputDC5V();
}

適配器類

public class PowerAdapter extends AC220 implements DC5{

@Override
public int outputDC5V() {
int adapterInput=outputAC220V();
int adapterOutput = adapterInput/44;
System.out.println("使用PowerAdapter輸入AC:"+adapterInput+"V"+"輸出DC:"+adapterOutput+"V");
return adapterOutput;
}
}

測試代碼:

public class Test {

public static void main(String[] args) {
DC5 dc5=new PowerAdapter();
System.out.println(dc5.outputDC5V());

}
}

輸出結果

 

  可能很多人看到這就會問,這不就是裝飾者模式嗎?這個不就相當於擴展功能嗎,其實呢這兩個模式所處的階段不同,一個是在軟體設計的時候需要考慮的,這個呢是在軟體後續維護的時候所考慮的。第二個其實就是最大的區別:裝飾者和被裝飾者之間的介面相同,而適配器和被適配器之間的介面是不相同的(當然有些特殊情況是相同的)。假如我是另外一個電子設備,我需要3V的介面,那麼我們只需要再定義一個DC3介面,修改PowerAdapter裡面的適配方法。但是我這個廠家並不再需要5V的介面了,所以我這邊有的其實只是DC3不是DC5.看看現在的UML類圖

  還是和裝飾者有區別的,當然你要是通過裝飾者來實現上述功能一樣能實現
2.對象適配器
  上述UML圖中我們可以看出AC220和PowerAdapter連接太過緊密了,如果我們這是個很複雜的系統,那這樣做無疑讓我們的適配器載入會很慢,畢竟子類要想初始化完,就必須要父類先初始化完,所以我們可以不用繼承,而使用成員變數的方式來解決,即修改PowerAdapter的代碼

public class PowerAdapter implements DC5,DC3{
private AC220 ac220=new AC220();
private int adapterInput=ac220.outputAC220V();
@Override
public int outputDC5V() {

int adapterOutput = adapterInput/44;
System.out.println("使用PowerAdapter輸入AC:"+adapterInput+"V"+"輸出DC:"+adapterOutput+"V");
return adapterOutput;
}

@Override
public int outputDC3V() {
int adapterOutput=adapterInput/73;
System.out.println("使用PowerAdapter輸入AC:"+adapterInput+"V"+"輸出DC:"+adapterOutput+"V");
return adapterOutput;
}
}

UML類圖

  從繼承變成了組合,這就是對象適配
3.JDK解讀
  XmlAdapter就是一個最典型的適配器,下麵我們來看代碼具體分析

定義一個學生類,將學生類序列化成xml文件

import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
import java.util.Date;

@XmlType(propOrder={"id","name","birthDay"}) //指定序列成的xml節點順序
@XmlAccessorType(value=XmlAccessType.FIELD) //訪問類型改為欄位
@XmlRootElement
public class Student {
@XmlElement
private String id;
@XmlElement
private String name;
@XmlJavaTypeAdapter(value=DateAdapter.class)
@XmlElement
private Date birthDay;

public String getId() {
return id;
}

public void setId(String id) {
this.id = id;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Date getBirthDay() {
return birthDay;
}

public void setBirthDay(Date birthDay) {
this.birthDay = birthDay;
}

@Override
public String toString() {
return "Student{" +
"id='" + id + '\'' +
", name='" + name + '\'' +
", birthDay=" + birthDay +
'}';
}
}

Date適配器

import java.text.SimpleDateFormat;
import java.util.Date;

import javax.xml.bind.annotation.adapters.XmlAdapter;

public class DateAdapter extends XmlAdapter<String, Date> {
//反序列化成日期對象Date
@Override
public Date unmarshal(String str) throws Exception {
SimpleDateFormat format = getSimpleDateFormat("yyyy-MM HH:mm:ss");
return str==null ? null:format.parse(str);
}
//序列化成xmL
@Override
public String marshal(Date date) throws Exception {
SimpleDateFormat format = getSimpleDateFormat("yyyy-MM HH:mm:ss");
return date==null ? "":format.format(date);
}
private SimpleDateFormat getSimpleDateFormat(String pattern){
SimpleDateFormat format = new SimpleDateFormat(pattern);
return format;
}
}

編寫測試類

import javax.xml.bind.JAXBContext;

import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
import java.io.*;
import java.util.Date;

public class Test {

public static void main(String[] args) {
DateAdapter da=new DateAdapter();
Student stu = new Student();
stu.setId("1");
stu.setName("方塊人");
stu.setBirthDay(new Date());
try {
JAXBContext context = JAXBContext.newInstance(Student.class);
Marshaller marshaller = context.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.setProperty(Marshaller.JAXB_ENCODING, "UTF-8");
//marshaller.marshal(stu, System.out);
StringWriter writer = new StringWriter();
marshaller.marshal(stu, writer);
System.out.println(writer.toString());
//反序列化
Unmarshaller unmarshaller = context.createUnmarshaller();
StringReader reader = new StringReader(writer.toString());
Student stu2 = (Student) unmarshaller.unmarshal(reader);

} catch (Exception e) {
e.getMessage();
e.printStackTrace();
}
}
}

輸出結果:

  註意看birthDay這個Date類型,我們已經在轉化成xml文件的時候將日期格式變化成功,如果是沒有進行日期適配器轉換的話的輸出結果是

  這樣就不是我們想要的格式

XmlAdapter源碼:

public abstract class XmlAdapter<ValueType,BoundType> {

/**
* Do-nothing constructor for the derived classes.
*/
protected XmlAdapter() {}

/**
* Convert a value type to a bound type.
*
* @param v
* The value to be converted. Can be null.
* @throws Exception
* if there's an error during the conversion. The caller is responsible for
* reporting the error to the user through {@link javax.xml.bind.ValidationEventHandler}.
*/
public abstract BoundType unmarshal(ValueType v) throws Exception;

/**
* Convert a bound type to a value type.
*
* @param v
* The value to be convereted. Can be null.
* @throws Exception
* if there's an error during the conversion. The caller is responsible for
* reporting the error to the user through {@link javax.xml.bind.ValidationEventHandler}.
*/
public abstract ValueType marshal(BoundType v) throws Exception;
}

  有兩個方法需要實現,一個是反序列化,另外一個是序列化。相信你看懂了適配器模式之後,也能理解這個類中的方法含義

總結:

  適配器模式更多的是提供不同介面給不同的廠家,適配器我們知道怎麼實現之後,更多的是在適用場景中去思考。這樣就會對適配器模式有更加深刻的理解。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、概述 AIDL是Android Interface Definition Language的縮寫,即Android介面定義語言。它是Android的進程間通信比較常用的一種方式。 Android中,每一個進程都有自己的Dalvik VM實例,擁有自己的獨立的記憶體空間,進程與進程之間不共用記憶體,這 ...
  • 今天狀態不太好,睡久了懵一天。 以前只是瞭解過async函數,並還沒有很熟練的運用過,所以先開個坑吧,以後再結合實際來更新下,可能說的有些問題希望大家指出。 async和await相信大家應該不陌生,讓非同步處理變得更友好。 其實這玩意兒就是個Generator的語法糖,想深入學習得去看看Genera ...
  • <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <script> var now = new Date; var bir = new Date(1995,09,24,08,50,10); //在這裡按照這個格式輸入自己生日 ...
  • vue+element ui中的圖片獲取與上傳 工作上接觸了一下圖片的處理,圖片的格式是文件流, 記錄如下。 請求圖片 請求圖片的時候,帶上 , 否則圖片顯示的可能是亂碼。 顯示圖片 圖片返回的是文件流的形式, 控制臺中顯示的是亂碼。 直接顯示二進位圖片會出錯,所以我們要進行處理。 顯示圖片中,要對 ...
  • 1.CSS認識 在談論CSS的概念之前,我們先說一說web標準的目的——其在於創建一個統一的用於web表現層的技術標準,以便通過不同瀏覽器或終端設備向最終用戶展示信息內容。一個網頁的呈現是由三部分組成:結構(Structure)、表現(Presentation)和行為(Behavior)。 而三大部 ...
  • 1、首先得有node和npm的環境,node的下載:http://nodejs.org/download/。安裝node之後,npm也自動生成了,顯示版本號就意味著安裝成功 2、接下來就是安裝vue-cli腳手架,執行以下命令: 在瀏覽器執行 http://localhost:8080,出現以下截圖 ...
  • 一、2D轉換(transform) CSS3中的transform轉換和PS中的變換是一樣的,分別有:縮放、位移、斜切、旋轉 1.1 transform:scale()縮放 transform:scale(w,h); 寬度和高度,w,h都是填寫縮放的倍數,沒有單位,比如1.5就是放大1.5倍,1是默 ...
  • 今天和大家分享下better-scroll這款移動端用來解決各種滾動需求的插件(目前已經支持PC) 關於其中的API大家可以去官網看下 這裡就給大家介紹幾種常用的以及需要註意的點是什麼 首先說一下better-scroll的使用註意問題吧 1、移動端 我們通常採用三段式進行佈局的 例如 <div c ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...