适配器模式
1. 概念
适配器模式是两个不兼容的接口之间的桥梁,其将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。
这种模式涉及到一个单一的类,该类负责加入独立的或不兼容的接口功能。比如,读卡器即是内存卡和笔记本之间的适配器,你将内存卡插入读卡器,再将读卡器插入笔记本,最终就可以通过笔记本来读取内存卡。
适配器的特点:
- 适配器通过继承或依赖已有的对象,实现想要的目标接口。
- 由于 JAVA 至多继承一个类,所以至多只能适配一个适配者类,而且目标类必须是抽象类。
- 过多地使用适配器,会让系统非常零乱,不易整体进行把握。比如,明明看到调用的是 A 接口,其实内部被适配成了 B 接口的实现,一个系统如果太多出现这种情况,无异于一场灾难。因此如果不是很有必要,可以不使用适配器,而是直接对系统进行重构。
适配器模式的类结构:

- Target(目标接口):所要转换为的、所期待的接口
- Adaptee(源接口):需要做适配的接口
- Adapter(适配器):将源接口适配成目标接口——继承源接口,实现目标接口
Java I/O中的适配器模式:
- 适配器:
InputStreamReader
- 源角色:
InputStream
- 目标接口:
Reader
2. 示例
我们通过下面的实例来演示适配器模式的使用。其中,音频播放器设备只能播放 mp3 文件,通过使用一个更高级的音频播放器来播放 vlc 和 mp4 文件。
我们有一个 MediaPlayer
接口和一个实现了 MediaPlayer
接口的实体类 AudioPlayer
,默认情况下,AudioPlayer
可以播放 mp3 格式的音频文件。同时,我们还有另一个接口 AdvancedMediaPlayer
和实现了 AdvancedMediaPlayer
接口的实体类,该类可以播放 vlc 和 mp4 格式的文件。
现在,我们想要让 AudioPlayer
播放其他格式的音频文件。为了实现这个功能,我们需要创建一个实现了 MediaPlayer
接口的适配器类 MediaAdapter
,并使用 AdvancedMediaPlayer
对象来播放所需的格式。
AudioPlayer
使用适配器类 MediaAdapter
传递所需的音频类型,不需要知道能播放所需格式音频的实际类。 AdapterPatternDemo
类使用 AudioPlayer
类来播放各种格式。

为媒体播放器和更高级的媒体播放器创建接口
public interface MediaPlayer { public void play(String audioType, String fileName); }
public interface AdvancedMediaPlayer { public void playVlc(String fileName); public void playMp4(String fileName); }
创建实现了
AdvancedMediaPlayer
接口的实体类public class VlcPlayer implements AdvancedMediaPlayer{ @Override public void playVlc(String fileName) { System.out.println("Playing vlc file. Name: "+ fileName); } @Override public void playMp4(String fileName) { //什么也不做 } }
public class Mp4Player implements AdvancedMediaPlayer{ @Override public void playVlc(String fileName) { //什么也不做 } @Override public void playMp4(String fileName) { System.out.println("Playing mp4 file. Name: "+ fileName); } }
创建实现了
MediaPlayer
接口的适配器类public class MediaAdapter implements MediaPlayer { AdvancedMediaPlayer advancedMusicPlayer; public MediaAdapter(String audioType){ if(audioType.equalsIgnoreCase("vlc") ){ advancedMusicPlayer = new VlcPlayer(); } else if (audioType.equalsIgnoreCase("mp4")){ advancedMusicPlayer = new Mp4Player(); } } @Override public void play(String audioType, String fileName) { if(audioType.equalsIgnoreCase("vlc")){ advancedMusicPlayer.playVlc(fileName); }else if(audioType.equalsIgnoreCase("mp4")){ advancedMusicPlayer.playMp4(fileName); } } }
创建实现了
MediaPlayer
接口的实体类public class AudioPlayer implements MediaPlayer { MediaAdapter mediaAdapter; @Override public void play(String audioType, String fileName) { //播放 mp3 音乐文件的内置支持 if(audioType.equalsIgnoreCase("mp3")){ System.out.println("Playing mp3 file. Name: "+ fileName); } //mediaAdapter 提供了播放其他文件格式的支持 else if(audioType.equalsIgnoreCase("vlc") || audioType.equalsIgnoreCase("mp4")){ mediaAdapter = new MediaAdapter(audioType); mediaAdapter.play(audioType, fileName); } else{ System.out.println("Invalid media. "+ audioType + " format not supported"); } } }
使用
AudioPlayer
来播放不同类型的音频格式public class AdapterPatternDemo { public static void main(String[] args) { AudioPlayer audioPlayer = new AudioPlayer(); audioPlayer.play("mp3", "beyond the horizon.mp3"); audioPlayer.play("mp4", "alone.mp4"); audioPlayer.play("vlc", "far far away.vlc"); audioPlayer.play("avi", "mind me.avi"); } }
执行程序,输出结果: Playing mp3 file. Name: beyond the horizon.mp3 Playing mp4 file. Name: alone.mp4 Playing vlc file. Name: far far away.vlc Invalid media. avi format not supported