适配器模式
一、适配器模式简介
意图:将一个类的接口转换成客户希望的另外一个接口。适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。
主要解决:主要解决在软件系统中,常常要将一些"现存的对象"放到新的环境中,而新环境要求的接口是现对象不能满足的。
何时使用: 1.系统需要使用现有的类,而此类的接口不符合系统的需要。 2.想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作,这些源类不一定有一致的接口。 3.通过接口转换,将一个类插入另一个类系中
如何解决:继承或依赖(推荐)。
关键代码:适配器继承或依赖已有的对象,实现想要的目标接口。
优点: 1.可以让任何两个没有关联的类一起运行。 2.提高了类的复用。 3.增加了类的透明度。 4.灵活性好。
缺点: 1.过多地使用适配器,会让系统非常零乱,不易整体进行把握,如果不是很有必要,可以不使用适配器,而是直接对系统进行重构。 2.由于 JAVA 至多继承一个类,所以至多只能适配一个适配者类,而且目标类必须是抽象类。
使用场景:有动机地修改一个正常运行的系统的接口,这时应该考虑使用适配器模式。
注意事项:适配器不是在详细设计时添加的,而是解决正在服役的项目的问题。
二、实现Demo
现状:有一个 MediaPlayer 接口和一个实现了 MediaPlayer 接口的实体类 AudioPlayer。默认情况下,AudioPlayer 只可以播放 mp3 格式的音频文件。还有另一个接口 AdvancedMediaPlayer 和实现了 AdvancedMediaPlayer 接口的实体类。该类可以播放 vlc 和 mp4 格式的文件。 目的:想要让 AudioPlayer 播放其他格式的音频文件。 实现:使用适配器模式,需要创建一个实现了 MediaPlayer 接口的适配器类 MediaAdapter,并使用 AdvancedMediaPlayer 对象来播放所需的格式。AudioPlayer 使用适配器类 MediaAdapter 传递所需的 音频类型,不需要知道能播放所需格式音频的实际类。
interface AdvancedMediaPlayer { public void playVlc(String name); public void playMp4(String name); } class VlcPlayer implements AdvancedMediaPlayer { @Override public void playVlc(String name) { System.out.println("play: vlc, name: " + name); } @Override public void playMp4(String name) { // do nothing } } class Mp4Player implements AdvancedMediaPlayer { @Override public void playVlc(String name) { // do nothing } @Override public void playMp4(String name) { System.out.println("play: mp4, name: " + name); } } interface MediaPlayer { public void play(String type, String name); } class MadiaAdapter implements MediaPlayer { AdvancedMediaPlayer amp; public MadiaAdapter(String type) { if (type.equalsIgnoreCase("mp4")) { amp = new Mp4Player(); } else if (type.equalsIgnoreCase("vlc")) { amp = new VlcPlayer(); } } @Override public void play(String type, String name) { if (type.equalsIgnoreCase("mp4")) { amp.playMp4(name); } else if (type.equalsIgnoreCase("vlc")) { amp.playVlc(name); } } } class AudioPlayer implements MediaPlayer { @Override public void play(String type, String name) { if (type.equalsIgnoreCase("mp3")) { System.out.println("play: mp3, name: " + name); } else if (type.equalsIgnoreCase("mp4") || type.equalsIgnoreCase("vlc")) { MediaPlayer mp1 = new MadiaAdapter(type); mp1.play(type, name); } else { System.out.println("AudioPlayer: not support type " + type); } } } public class AdapterPatternDemo { public static void main(String args[]) { MediaPlayer p1 = new AudioPlayer(); p1.play("mp3", "hello.mp3"); p1.play("vlc", "dump.vlc"); p1.play("mp4", "run.mp4"); } }
参考:http://www.runoob.com/design-pattern/adapter-pattern.html
posted on 2019-04-10 18:14 Hello-World3 阅读(128) 评论(0) 编辑 收藏 举报
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!