手机操作问题
手机有不同的样式和品牌,每一种样式下面有各种品牌,每一种品牌也有多种样式,当然随时可能增加或者减少某一种样式或者某一种品牌的手机,需要考虑ocp原则,设计方案时;
传统方案解决手机使用问题(类图):
传统方案解决手机操作问题
- 扩展性问题(类爆炸),如果我们再增加手机的样式(旋转式),就需要增加各个品牌手机的类,同样如果我们增加一个手机品牌,也要在各个手机样式类下增加。
- 违反了单一职责原则,当我们增加手机样式时,要同时增加所有品牌的手机,这样增加了代码维护成本.
- 解决方案-使用桥接模式
桥接模式(Bridge)
基本介绍
- 桥接模式(Bridge模式)是指:将 实现与 抽象 放在两个不同的类层次中,使两个层次可以独立改变。抽象聚合了实现
- 是一种结构型设计模式
- Bridge模式基于类的最小设计原则,通过使用封装、聚合及继承等行为让不同
的类承担不同的职责。它的主要特点是把抽象(Abstraction)与行为实现(Implementation)分离开来,从而可以保持各部分的独立性以及应对他们的功能扩展,对于哪个是抽象,哪个是实现是自己需要分辨的;
原理类图
桥接模式解决手机操作问题
这里选取了手机(Phone)作为抽象,品牌(Brand)作为实现来进行设计的,同时将Brand聚合到Phone里面了;
//接口 实现方
public interface Brand {
void open();
void close();
void call();
}
//具体的实现方1
public class XiaoMi implements Brand {
@Override
public void open() {
System.out.println(" 小米手机开机 ");
}
@Override
public void close() {
System.out.println(" 小米手机关机 ");
}
@Override
public void call() {
System.out.println(" 小米手机打电话 ");
}
}
//具体的实现方2
public class Vivo implements Brand {
@Override
public void open() {
System.out.println(" Vivo手机开机 ");
}
@Override
public void close() {
System.out.println(" Vivo手机关机 ");
}
@Override
public void call() {
System.out.println(" Vivo手机打电话 ");
}
}
//抽象方
public abstract class Phone {
//聚合品牌
private Brand brand;
//构造器
public Phone(Brand brand) {
super();
this.brand = brand;
}
protected void open() {
this.brand.open();//抽象Phone作为桥的实现:调用某个品牌的open方法,因为聚合了实现
}
protected void close() {
brand.close();
}
protected void call() {
brand.call();
}
}
//抽象方实现1
//折叠式手机类,继承 抽象类 Phone
public class FoldedPhone extends Phone {
//构造器
public FoldedPhone(Brand brand) {
super(brand);
}
public void open() {
super.open();
System.out.println(" 折叠样式手机 ");
}
public void close() {
super.close();
System.out.println(" 折叠样式手机 ");
}
public void call() {
super.call();
System.out.println(" 折叠样式手机 ");
}
}
//抽象方实现2 新增加一个直立样式的手机很方便
public class UpRightPhone extends Phone {
//构造器
public UpRightPhone(Brand brand) {
super(brand);
}
public void open() {
super.open();//这里调用了Brand某个品牌的open方法
System.out.println(" 直立样式手机 ");
}
public void close() {
super.close();
System.out.println(" 直立样式手机 ");
}
public void call() {
super.call();
System.out.println(" 直立样式手机 ");
}
}
//client调用
public class Client {
public static void main(String[] args) {
//获取折叠式手机 (样式 + 品牌 )
Phone phone1 = new FoldedPhone(new XiaoMi()); //抽象方聚合了实现方
phone1.open();//手机phone1的open实际上是通过FoldedPhone具体的抽象作为桥,进而调用到XiaoMi的open方法
phone1.call();
phone1.close();
System.out.println("=======================");
Phone phone2 = new FoldedPhone(new Vivo());//抽象方聚合了实现方
phone2.open();
phone2.call();
phone2.close();
System.out.println("==============");
UpRightPhone phone3 = new UpRightPhone(new XiaoMi());//抽象方聚合了实现方
phone3.open();
phone3.call();
phone3.close();
System.out.println("==============");
UpRightPhone phone4 = new UpRightPhone(new Vivo());//抽象方聚合了实现方
phone4.open();
phone4.call();
phone4.close();
}
}
类图:
分析:这个类图Phone作为抽象,Brand作为实现,Phone作为桥接模式中的桥的作用;比如说new UpRightPhone(new Vivo())的call()方法的调用:直立手机的call方法进而调用父类Phone的open,再而调用了Vivo手机的call方法,这里Phone就是桥的实现,直立手机+vivo(样式+品牌)都经过了;
桥接模式在JDBC 的源码剖析
- Jdbc 的 Driver接口,如果从桥接模式来看,Driver就是一个接口,下面可以有MySQL的Driver,Oracle的Driver,这些就可以当做实现接口类,这里的抽象是DriverManager(不是抽象类,不是非常标准的桥接模式)
- 代码分析+Debug源码
DriverManager
获取的Connection有Mysql,Oracle等
桥接模式的注意事项和细节 - 实现了抽象和实现部分的分离,从而极大的提供了系统的灵活性,让抽象部分和实现部分独立开来,这有助于系统进行分层设计,从而产生更好的结构化系统。
- 对于系统的高层部分,只需要知道抽象部分和实现部分的接口就可以了,其它的部分由具体业务来完成。
- 桥接模式替代多层继承方案,可以减少子类的个数,降低系统的管理和维护成本。
- 桥接模式的引入增加了系统的理解和设计难度,由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计和编程
- 桥接模式要求正确识别出系统中两个独立变化的维度,正确识别出哪个是抽象,哪个是实现,哪个作为桥比较好??,因此其使用范围有一定的局限性,即需要有下面的应用场景。
桥接模式其它应用场景
- 对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用.
- 常见的应用场景:
- JDBC驱动程序
- 银行转账系统
转账分类: 网上转账,柜台转账,AMT转账 ---------抽象
转账用户类型:普通用户,银卡用户,金卡用户… -----实现 - 消息管理
消息类型:即时消息,延时消息 ---------抽象
消息分类:手机短信,邮件消息,QQ消息… -----实现
来源:CSDN
作者:奋斗小亮
链接:https://blog.csdn.net/weixin_43719015/article/details/104572795