Java设计模式之桥接模式
作者:tianClassmate 发布时间:2021-08-31 05:56:02
本文通过老王和小王买车,引出设计模式中的结构型设计之桥接模式,接着说明设计型模式的概念和代码实现,为了加深理解,会说明适配器设计模式在JDBC中的应用,最后谈谈桥接模式和适配器模式的总结。
读者可以拉取完整代码到本地进行学习,实现代码均测试通过后上传到码云,本地源码下载。
一、引出问题
老王和小王去奔驰4S店买车,奔驰4S店的各种品牌型号琳琅满目,老王想试驾奔驰E、小王想试驾奔驰G,并且提出两种奔驰型号的各种颜色都想体验一把,这让店小二犯了难,两两组合就是很多种,4S店压根放不下。
无奈店小二求救经理,经理出了一个注意:将奔驰E和G开的品牌抽象出来,将颜色也抽象出来,通过品牌和颜色的组合代替继承关系,减少了颜色和品牌的耦合,且减少了车的个数,只需要两台就够了。
果然经理不愧是经理。
经理所说的其实就是桥接模式。这种模式涉及到一个作为桥接的接口,使得实体类的功能独立于接口实现类。这两种类型的类可被结构化改变而互不影响。
二、概念与使用
我们看一些概念:桥接(Bridge)是用于把抽象化与实现化解耦,使得二者可以独立变化。这种类型的设计模式属于结构型模式,它通过提供抽象化和实现化之间的桥接结构,来实现二者的解耦。
在该模式中应该涉及到四个角色:
①实现类接口(Implementor):定义实现角色的接口,供扩展抽象化角色使用,例如抽象出奔驰品牌benz 可以扩展出 benzE benzG
②具体实现角色(ConcreteImplementor):实现类的具体实现,例如各种奔驰品牌
③抽象化(Abstraction)角色:定义一个抽象类,其中引用了实现化角色(想要组合),例如汽车产品
④扩展抽象化(RefinedAbstraction)角色:抽象化角色子类,实现父类方法,且通过组合关系调用实现化角色中的业务方法,例如具体奔驰产品,红色奔驰、白色奔驰
根据该模式的定义,我们将奔驰品牌抽象出来,然后各品牌有各自的实现,每个颜色的车把车品牌组合进来,在客户端中每个相机类型和相机品牌都能两两组合。
我们看具体的代码实现:
实现类接口:
/**
* 奔驰品牌类
* @author tcy
* @Date 05-08-2022
*/
public interface BenzBrand {
void showInfo();
}
具体实现角色1:
/**
* @author tcy
* @Date 05-08-2022
*/
public class BenzE implements BenzBrand{
@Override
public void showInfo() {
System.out.print("【奔驰E】颜色是:");
}
}
具体实现角色2:
/**
* @author tcy
* @Date 05-08-2022
*/
public class BenzG implements BenzBrand{
@Override
public void showInfo() {
System.out.print("【奔驰G】颜色是:");
}
}
抽象化角色:
/**
* 抽象奔驰类
* @author tcy
* @Date 05-08-2022
*/
public abstract class Benz {
// 将品牌组合进来
protected BenzBrand benzBrand;
public Benz(BenzBrand benzBrand) {
this.benzBrand = benzBrand;
}
public void showInfo(){
benzBrand.showInfo();
}
}
扩展抽象化1:
/**
* @author tcy
* @Date 05-08-2022
*/
public class BlackBenz extends Benz {
public BlackBenz(BenzBrand benzBrand) {
super(benzBrand);
}
@Override
public void showInfo() {
super.showInfo();
System.out.println("黑色...");
}
}
扩展抽象化2:
/**
* @author tcy
* @Date 05-08-2022
*/
public class RedBenz extends Benz {
public RedBenz(BenzBrand benzBrand) {
super(benzBrand);
}
@Override
public void showInfo() {
super.showInfo();
System.out.println("红色...");
}
}
客户端调用:
/**
* @author tcy
* @Date 05-08-2022
*/
public class Client {
public static void main(String[] args) {
// 黑色奔驰E
Benz benz1 = new BlackBenz(new BenzE());
benz1.showInfo();
// 黑色奔驰G
Benz benz2 = new BlackBenz(new BenzG());
benz2.showInfo();
// 红色奔驰E
Benz benz3 = new RedBenz(new BenzE());
benz3.showInfo();
// 红色奔驰G
Benz benz4 = new RedBenz(new BenzG());
benz4.showInfo();
}
}
【奔驰E】颜色是:黑色...
【奔驰G】颜色是:黑色...
【奔驰E】颜色是:红色...
【奔驰G】颜色是:红色...
这样即使老王提出来新的颜色、新的车型,只需要增加相应的具体实现角色或者扩展抽象化角色即可。
顾名思义,桥接模式就像是一个桥,可以用来连接两个不同地方,这两个地方自由发展,中间的贸易是通过一座桥来连接。
这种方法的缺点也很显著,汽车能很快的确立型号和颜色两个维度,在实际业务开发中,识别出系统两个独立变化的维度就不简单了。
不难看出,列举的例子有些过于强求,在现实世界中是永远不可能发生的,为了加深理解我找了大量在JDK亦或是Spirng等各种框架对桥接模式的应用,只找到了桥接模式在Jdbc中的应用。
三、应用
我们都知道通过JDBC可以完成Java对关系型数据库的SQL操作,我们在连接数据数据库时,想必都接触过Driver,在连接MySQL和Oracle的Driver都是不同的,这些都是实现接口类。
我们看一下MySQL中实现的Driver类。
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
public Driver() throws SQLException {
}
static {
try {
DriverManager.registerDriver(new Driver());
} catch (SQLException var1) {
throw new RuntimeException("Can't register driver!");
}
}
}
在该类中实际上有两个作用,一是调用了DriverManager中的registerDriver方法来注册驱动,二是当驱动注册完成后,我们就会开始调用DriverManager中的getConnection方法了。
我们看DriverManager的完整代码:
public class DriverManager {
public static Connection getConnection(String url,
String user, String password) throws SQLException {
java.util.Properties info = new java.util.Properties();
if (user != null) {
info.put("user", user);
}
if (password != null) {
info.put("password", password);
}
return (getConnection(url, info, Reflection.getCallerClass()));
}
private static Connection getConnection(
String url, java.util.Properties info, Class<?> caller) throws SQLException {
/*
* When callerCl is null, we should check the application's
* (which is invoking this class indirectly)
* classloader, so that the JDBC driver class outside rt.jar
* can be loaded from here.
*/
ClassLoader callerCL = caller != null ? caller.getClassLoader() : null;
synchronized(DriverManager.class) {
// synchronize loading of the correct classloader.
if (callerCL == null) {
callerCL = Thread.currentThread().getContextClassLoader();
}
}
if(url == null) {
throw new SQLException("The url cannot be null", "08001");
}
println("DriverManager.getConnection(\"" + url + "\")");
// Walk through the loaded registeredDrivers attempting to make a connection.
// Remember the first exception that gets raised so we can reraise it.
SQLException reason = null;
for(DriverInfo aDriver : registeredDrivers) {
// If the caller does not have permission to load the driver then
// skip it.
if(isDriverAllowed(aDriver.driver, callerCL)) {
try {
println(" trying " + aDriver.driver.getClass().getName());
Connection con = aDriver.driver.connect(url, info);
if (con != null) {
// Success!
println("getConnection returning " + aDriver.driver.getClass().getName());
return (con);
}
} catch (SQLException ex) {
if (reason == null) {
reason = ex;
}
}
} else {
println(" skipping: " + aDriver.getClass().getName());
}
}
// if we got here nobody could connect.
if (reason != null) {
println("getConnection failed: " + reason);
throw reason;
}
println("getConnection: no suitable driver found for "+ url);
throw new SQLException("No suitable driver found for "+ url, "08001");
}
}
}
在Java中通过Connection提供给各个数据库一样的操作接口,这里的Connection可以看作抽象类。
可以说我们用来操作不同数据库的方法都是相同的,不过MySQL有自己的ConnectionImpl类,同样Oracle也有对应的实现类。
这里Driver和Connection之间是通过DriverManager类进行桥接的,这种桥接模式和我们上面可以清晰的看出来各个角色是不同的。
四、总结
桥接模式是很好理解的,相信认真看了实例的同学应该都可以看懂,但那并不代表你已经掌握了该设计模式。在我们使用JDBC的时候,想必有很多同学并不能看出来这是桥接模式。
纸上得来终觉浅,有一部分例子是为了说明桥接模式而“构想”出来的,各个角色都是清晰直观。看了这样的代码,可以学会桥接模式,但是到了实际中很可能还是不会用。
最好的方法就是给出真实项目里的例子。但是这个难度确实很大,一到了真实项目里,就会遇到很多细节问题,从而影响对模式的理解,而且真实项目都带有一定的业务环境。
看懂并且学会了设计模式是一回事,在实际开发中择优选择设计模式那是另外一回事,这不仅需要对各个设计模式理解到位,更多的是对业务的理解和代码理念的把控。
来源:https://www.cnblogs.com/tianClassmate/p/16554607.html


猜你喜欢
- 一、方法(Method)概念 1、Java 中的方法就是其他编程语言中的函数(Function) 2、方法的定义格式:①
- XML对开发者来说十分的方便,不仅使用起来简单,而且能够及时调试,修改界面之后马上能看到效果。Java设置布局不具有这个优势。但是java却
- 1、final修饰类被final修饰的类不能被继承,因此final类的成员方法也不能被覆写,被final关键字修饰的类没有子类,因此类的实现
- 序言for循环语句是java循环语句中最常用的循环语句,一般用在循环次数已知的情况下使用。for循环语句的语法格式如下:java语言 for
- 上一小节我们学习了FastThreadLocal的创建和get方法的实现逻辑, 这一小节学习FastThreadLocal的set方法的实现
- 现在,汽车的踪影无处不在,公路上疾驰,大街边临停,小区中停靠,车库里停泊。管理监控如此庞大数量的汽车是个头疼的问题。精明的人们把目光放在车牌
- 旋转扭曲特效是指在一个圆形区域内扭曲所渲染的图像,其他像素的旋转程度随着距离的变化而变化。具体可以通过修改Shader来实现。原始图片扭曲图
- 实现消息队列的两种方式Apache ActiveMQ官方实例发送消息直接在Apache官网http://activemq.apache.or
- 第一种查看防火墙是否打开6379端口 查看防火墙状态systemctl status firewalld如果防火墙没有启动的话。可以选择直接
- 赋值运算符也有和算数操作符所结合的用法之前附录中有提及,用法是:比如要将x加上4,然后再赋值给x,就可以写成x+=4. public cla
- 我们已经写了一些Java程序。之前的每个Java程序都被保存为一个文件,比如Test.java。随后,该程序被编译为Test.class。我
- 简述偶然看到一篇关于阿里新orm框架的文章,好奇的点了进去。开发后端多年,看到这个还是有点兴奋的。常用mysql的orm框架mybatis、
- 项目地址:https://github.com/JeasonWong/SlackLoadingView老规矩,先上效果。图好大。。说下第一眼
- 一、简介Lock关键字是Monitor的一种替换用法,lock在IL代码中会被翻译成Monitor. lock (obj) &nb
- 1. 原因最近学习spring data JPA 时候要用到分页功能,但是发现网上所有教程都是通过new PageRequest()方法解决
- 介绍:淡入淡出动画(也称为“叠化”)逐渐淡出一个 View 或 ViewGroup,同时淡入另一个。此动画适用于您希望在应用中切换内容或视图
- 本文实例讲述了Android中Java根据文件头获取文件类型的方法。分享给大家供大家参考,具体如下:前面讲过Android系统内部的Medi
- Spring AOP proxyTargetClass的行为要点列表形式proxyTargetClasstrue目标对象实现了接口 – 使用
- 本文实例讲述了C#使用HttpDownLoadHelper下载文件的方法。分享给大家供大家参考。具体实现方法如下:using System;
- 这篇文章主要介绍了SpringBoot+SpringCloud实现登录用户信息在微服务之间的传递,文中通过示例代码介绍的非常详细,对大家的学