1、桥接模式
桥接(Bridge)模式包含以下主要角色。
- 抽象化(Abstraction)角色:定义抽象类,并包含一个对实现化对象的引用。
- 扩展抽象化(Refined Abstraction)角色:是抽象化角色的子类,实现父类中的业务方法,并通过组合关系调用实现化角色中的业务方法。
- 实现化(Implementor)角色:定义实现化角色的接口,供扩展抽象化角色调用。
- 具体实现化(Concrete Implementor)角色:给出实现化角色接口的具体实现。
桥接模式,是在一个类中,某个对象有不同维度的实现时使用
比如
示例:按用途分可选钱包(Wallet)和挎包(HandBag),按颜色分可选黄色(Yellow)和红色(Red)。可以按两个维度定义为颜色类和包类。
2、适配器模式
特点:
将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。
该模式的主要优点如下。
- 客户端通过适配器可以透明地调用目标接口。
- 复用了现存的类,程序员不需要修改原有代码而重用现有的适配者类。
- 将目标类和适配者类解耦,解决了目标类和适配者类接口不一致的问题。
- 在很多业务场景中符合开闭原则。
缺点:
- 适配器编写过程需要结合业务场景全面考虑,可能会增加系统的复杂性。
- 增加代码阅读难度,降低代码可读性,过多使用适配器会使系统代码变得凌乱。
适配器有两种结构模式
1、类适配器
2、对象适配器
类适配器,是通过继承来实现的,而对象适配器,则是在适配器中引入一个对象的引用
示例:
比如,电脑的USB接口,需要连接MP3和MP4,但是MP3和MP4各自所需要实现的东西又不一样,所以对于适配这两样物品,USB接口是不管的,它只负责提供一个接口,用来连接电脑,具体做什么事情,由接入接口的设备来实现
代码如下:
首先创建一个USB接口,定义一个request()方法
package com.hzf.demo.adapter;
/**
* use标准接口,只提供接入电脑,不管插入的是读卡器还是耳机
*/
public interface USB {
public void request();
}
然后来编写适配器,首先创建一个MP3类,代表MP3这个物品
package com.hzf.demo.adapter;
public class MP3 {
public void map3PlayRequest(){
System.out.println("播放mp3");
}
}
再创建一个MP4类,代表MP4
package com.hzf.demo.adapter;
public class MP4 {
public void mp4PlayRequest(){
System.out.println("播放Mp4");
}
}
现在有了USB接口,有了MP3和MP4,就需要对他们进行适配
创建MP3适配器,实现USB接口,并创建一个MP3对象的引用
package com.hzf.demo.adapter;
public class Mp3Adapter implements USB{
MP3 mp3 = new MP3();
@Override
public void request() {
mp3.map3PlayRequest();
}
}
创建MP4适配器,实现USB接口,并创建一个MP4对象的引用
package com.hzf.demo.adapter;
public class Mp4Adapter implements USB{
MP4 mp4 = new MP4();
@Override
public void request() {
mp4.mp4PlayRequest();
}
}
编写测试类
package com.hzf.demo.adapter;
public class AdapterTest {
public static void main(String[] args) {
USB usb = new Mp3Adapter();
usb.request();
}
}
3、桥接模式结合适配器模式
简单来说,就是将桥接模式中抽离的抽象模块,套用适配器模式
例如:
现在有两台打印机,一台彩色打印机,一台黑白打印机,信息来源有两种方式,扫描仪和USB接口传输
1、定义一个数据收集接口
public interface CollectText {
String collect();
}
2、定义打印机抽象父类
public abstract class Printer {
public CollectText collectText;
private Printer(){
}
public Printer(CollectText collectText){
this.collectText = collectText;
}
public abstract void print();
}
3、创建彩色打印机,继承打印机父类
public class ColoursPrinter extends Printer {
public ColoursPrinter(CollectText collectText) {
super(collectText);
}
@Override
public void print() {
String collect = collectText.collect();
System.out.println("彩色打印机打印内容:"+collect);
}
}
4、创建黑白打印机,继承打印机父类
public class MonochromePrinter extends Printer {
public MonochromePrinter(CollectText collectText) {
super(collectText);
}
@Override
public void print() {
String collect = collectText.collect();
System.out.println("黑白打印机打印内容:"+collect);
}
}
5、套用适配器模式实现数据收集
扫描仪与USB接口收集数据本就是不兼容的,所以这里适合套用适配器模式进行兼容
这里使用对象适配器结构
6、创建扫描仪类
public class Scanner {
public String scan(){
return "扫描仪检测到图像。。。。。。";
}
}
7、创建扫描仪适配器
public class ScannerAdapter implements CollectText {
private Scanner scanner = new Scanner();
@Override
public String collect() {
return scanner.scan();
}
}
8、创建USB类
public class USB {
public String reader(){
return "USB读取到内容。。。。";
}
}
9、创建USB适配器
public class USBAdapter implements CollectText {
private USB usb = new USB();
@Override
public String collect() {
return usb.reader();
}
}
10、测试
public class BridgeAdapterTest {
public static void main(String[] args) {
//使用扫描仪收集内容,通过彩色打印机打印
CollectText collectText = new ScannerAdapter();
ColoursPrinter coloursPrinter = new ColoursPrinter(collectText);
coloursPrinter.print();
//使用USB收集内容,通过彩色打印机打印
collectText = new USBAdapter();
coloursPrinter = new ColoursPrinter(collectText);
coloursPrinter.print();
//使用扫描仪收集内容,通过黑白打印机打印
collectText = new ScannerAdapter();
MonochromePrinter monochromePrinter = new MonochromePrinter(collectText);
monochromePrinter.print();
//使用USB收集内容,通过黑白打印机打印
collectText = new USBAdapter();
monochromePrinter = new MonochromePrinter(collectText);
monochromePrinter.print();
}
}
11、结果
同理还有以下例子