本文通过老王和小王买车,引出设计模式中的结构型设计之桥接模式,接着说明设计型模式的概念和代码实现,为了加深理解,会说明适配器设计模式在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的时候,想必有很多同学并不能看出来这是桥接模式。

纸上得来终觉浅,有一部分例子是为了说明桥接模式而“构想”出来的,各个角色都是清晰直观。看了这样的代码,可以学会桥接模式,但是到了实际中很可能还是不会用。

最好的方法就是给出真实项目里的例子。但是这个难度确实很大,一到了真实项目里,就会遇到很多细节问题,从而影响对模式的理解,而且真实项目都带有一定的业务环境。

看懂并且学会了设计模式是一回事,在实际开发中择优选择设计模式那是另外一回事,这不仅需要对各个设计模式理解到位,更多的是对业务的理解和代码理念的把控。

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对Devmax的支持。如果你想了解更多相关内容请查看下面相关链接

Java设计模式之桥接模式的更多相关文章

  1. ios – 从Swift重写一个Obj-C类方法,返回NSArray

    .要在Swift子类中指定可选或具体值,请使用Objective-Cnullability说明符:exampleMethod;桥接以覆盖类funcexampleMethod()–>[AnyObject]exampleMethod;桥接以覆盖类funcexampleMethod()–>[AnyObject]?exampleMethod;和exampleMethod;具有相同的行为并桥接到隐式解包的可选项.

  2. 开始使用 swift 的 c语言 库

    为了手头上的一个项目,我需要使用CommonCrypto库中的HMAC函数.虽然苹果在swift中已经提供了许多系统库,但是CommonCrypto不在其中.庆幸的是,要使用这个库并不怎么费事,只需要做一点额外的工作.开始访问库在使用库之前,我们需要通知Swift编译器.要完成这个过程,我们有两种方式.它们都能在示例工程中正常运行,但是你应该根据你代码的用途来选择具体的方式.好消息是,你随便使用那

  3. Swift 与 Object-C 交互 Swift版本为:1.2

    一个是Swift项目调用Object-C的类,另一个是Object-C项目调用Swift类。我们的结构目录变成这样,我这里新建的桥接文件名为“SwiftCallObject-C-Bridging-Header.h”。如果你是第在Object-C项目里第一次创建Swift的类,那么依旧会有窗口提示,询问你是否需要创建桥接文件。这里先说明一下,在Object-C中调用Swift时,Xcode会帮助我们自动生成头文件,不需要我们去维护。谨记,在Object-C调用Swift端代码时,Swift类的class前需

  4. Swift和OC混用

    记得去年在北京一家公司的时候就使用了swift和OC混编。在这里想写一下swift和OC混用的一些简单使用方法,如果有错误请批评指正.在一个应用的target中导入一些Objective-C文件供Swift代码使用时,你需要依赖于Objective-C的桥接头文件来暴露给Swift。当你添加Swift文件到现有的Objective-C应用(或反之)时,Xcode会自动创建这些头文件。在这个桥接头文件中列出的所有public的Objective-C头文件都会对Swift可见。用Swift语法使用这些Obje

  5. swift开发

    [myTableViewinsertSubview:mySubviewatIndex:2在Swift中如下调用。在Objective-C中的指针在导入Swift时被映射为Swift语言的optional类型。importUIKitclassMySwiftViewControllerUIViewController{//definetheclass}如下定义了一个采用Objective-C中的UITableViewDelegate和UITableViewDataSource协议的Swift类MySwiftV

  6. Swift和OC文件间的相互调用

    公共文件/桥接的方法2016/1/2613:33Swift使用OC文件方法看完上面内容之后,你只需要这么做,swift就能使用OC的类了注意这里是搜索bridging注意这里是搜索bridgingOC使用Swift文件方法在你需要使用的文件中导入头文件#import"-Swift.h"如果不行,创建一个桥接文件就行,桥接文件内不写任何代码.Xcode路径要填#

  7. swift 与oc 混编

    生成一个桥接文件Swift_Objc-Bridging-Header.h,这个文件在swift调用oc时学要用到。在viewController,中引入Swift_Objc-Swift.h文件,即可用oc语言使用swift类ViewController.h文件中调用代码:swift调用oc:在Swift_Objc-Bridging-Header.h文件中引入.h头文件,即可用swift语言调用oc类。

  8. oc 和swift 混编之--准备工作

    之前的一段时间在学习swift,觉得可以出师了,哈哈哈,遂打算把oc和swift混编起来,混起来才发现远远比想象的要复杂。因为当前正在做的项目是用oc写的,也不打算用swift重新写一遍,所以直接在当前项目里加入swift代码,这当然是允许的,不过在此之前要做一些准备工作。也很简单,在你需要使用的那个类里边,import这样一个文件“项目名-Swift.h”:8E2FD4C0-4DF2-4A1B-BB66-9F0D43E5BF35.png这样oc就可以访问swift了。

  9. Swift与OC混合使用

    Swift中调用OC文件如果你是在原有的Swift项目中创建OC文件,那么Xcode将自动进行提示,询问你是否要建立一个桥接,然后选择是。如果是在Swift中引入一个OC文件的话,而之前你项目中又不存在这个桥接文件,那么你就要手动创建一个。OC文件调用Swift如果你的OC文件或者项目中需要用的Swift中的类,但是Swift中又不提供头文件怎么办,没关系,只需要在OC文件中引入”项目名-Swift.h“,即可成功调用。,如下图

  10. swift oc 双语言混编 持续更新

    过分依赖IDE带来的后果,就是一旦出现问题,根本就摸不清头脑,找不到根源!d.桥文件出现了,注意它目前的位置,是在Target包里,而不是在项目包里面!f.最后,在我们的.h桥接文件中,import想要在swift中用到的OC类就行了,大功告成!

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部