热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

设计模式深度解析:桥接模式的应用与实现

设计模式:桥接(Bridge)模式一、前言  写到这里,基本上就是对前面几种模式的扩展和区别了,可以看到我们前面的几种模式,很多时候都出现了重叠,这里要分清一个概念,模式并不是完全隔离和独立

设计模式:桥接(Bridge)模式

一、前言

  写到这里,基本上就是对前面几种模式的扩展和区别了,可以看到我们前面的几种模式,很多时候都出现了重叠,这里要分清一个概念,模式并不是完全隔离和独立的,有的模式内部其实用到了其他模式的技术,但是又有自己的创新点,如果一味地认为每个模式都是独一无二,与其他模式完全区别的,这是一种误区,因此,这里又看到了基础知识的重要性。从不同的角度都有各自的共同性,比如从人的角度,每个人都是不同的,可若是从原子和分子的角度,每个人又是相同的,那到底是相同还是不同,取决于我们看问题的角度。因此我们要学会求同辨异!桥接模式是一种将类的功能层次和实现层次分离的技术,所谓类的功能层次指的是类要实现什么功能,要定义多少个函数还进行处理,在功能之中我们会用到继承来定义新的方法同时也能使用父类的方法,这样就构成了一个层次“父类-子类-孙类...”,这就是功能层次,与之对应的就是实现层次了,其实也很好理解,如果我们事先确定了完成一件事情的最基本的子功能,那么我们定义这些子功能为接口或者抽象类,然后使用实现类来进行实现,这样一个抽象类,多个实现类,(抽象类——>(实现类1,实现类2,实现类...))这样的结构就是实现层次,可以看到高度一直为2,而功能层次高度为N(N为继承的层次)。那么可不可以将这两者分离呢,一方面我们知道一个类的基础子功能并且能够使用到这些子功能的实现,另一方面我们可以在这些子功能的基础上定义出我们自己需要的功能(功能层次),如果可以的话,基本的元素就相当于空气、水分等元素,而我们需要的功能就是将这些东西组成一种种不同的物质。这里就用到了委托,或者说是组合。实现了功能层次和实现层次分离的结构,我们称之为桥接模式。

技术分享图片

二、代码实现

   如上图所示,只有DisplayImpl是抽象类,其他都是具体类,其实DisplayImpl也可以改成接口,这在原理和理念上都是一致的。在实现层次,定义了三个rawXX元素,然后对这些元素进行实现(StringDisplayImpl),这样就保证了可扩展性,我们可以实现很多这样的类,但是高度一直都不变,是一个平行关系。对于功能层次,主要是对实现类中的元素进行操作,同样的使用方法对其进行简单的封装,便于子类的继承和使用,因为我们定义的displayImpl是私有的,这种组合或者说委托关系,我们在builder模式中肯定很熟悉,这里要说一些区别,首先是思想上的区别,采用桥接模式,就是要将功能层次和实现层次剥离的,因此功能层次上必须有继承,这样才有意义,另外,对于元素的组织上,builder是定义了一个操作,这个操作有顺序的对元素进行处理,无论实现层次上实现了多少个类,都要按照这个顺序来处理,而桥接模式,我们可以定义新的操作,可以灵活地使用元素,可以随意的增加新的功能,定义新的顺序,这是一个很大的不同;另外在于Main对类的使用上,对于builder模式,我们可能最后使用对应于实现层次上的一些方法来得到结果,而在桥接模式,我们本来就是面向接口(抽象)编程,因此代码中只是使用实现层次中的实现类new一个对象然后就可以使用了,没有这种复杂的关联关系。同时关于实现层次的实现类,为了很好的创造对象,我们可能使用抽象工厂模式来创造对象,需要根据实际情况来取舍。理解了这一点,我们来看代码。

 DisplayImpl类:虽然是实现层次,但却是抽象的。

1 package zyr.dp.bridge;
2
3 public abstract class DisplayImpl {
4 public abstract void rawOpen();
5 public abstract void rawPrint();
6 public abstract void rawClose();
7 }

 StringDisplayImpl类:

1 package zyr.dp.bridge;
2
3 public class StringDisplayImpl extends DisplayImpl {
4
5 String name;
6 public StringDisplayImpl(String name){
7 this.name=name;
8 }
9
10 public void rawOpen() {
11 printline();
12 }
13
14 public void rawPrint() {
15 System.out.println("|||||"+name+"|||||");
16 }
17
18 public void rawClose() {
19 printline();
20 }
21 private void printline(){
22 System.out.println("================");
23 }
24
25 }

 Display 类:

1 package zyr.dp.bridge;
2
3 public class Display {
4 private DisplayImpl displayImpl ;
5 public Display(DisplayImpl displayImpl){
6 this.displayImpl=displayImpl;
7 }
8 public void open(){
9 displayImpl.rawOpen();
10 }
11 public void print(){
12 displayImpl.rawPrint();
13 }
14 public void close(){
15 displayImpl.rawClose();
16 }
17 public final void display(){
18 open();
19 print();
20 close();
21 }
22
23 }

 CountDisplay类:

1 package zyr.dp.bridge;
2
3 public class CountDisplay extends Display {
4
5 public CountDisplay(DisplayImpl displayImpl) {
6 super(displayImpl);
7 }
8 public final void mutilDisplay(){
9 open();
10 for(int i=0;i<5;i++){
11 print();
12 }
13 close();
14 }
15
16 }

 Main类:

package zyr.dp.bridge;
public class Main { public static void main(String[] args) {
Display display
= new Display(new StringDisplayImpl("朱彦荣"));
display.display();
CountDisplay cd
= new CountDisplay(new StringDisplayImpl("李山秀"));
cd.display();
cd.mutilDisplay();
}
}

 结果:

技术分享图片

三、总结

  从上面我们可以看到这种方式的实现的好处了,首先如果我们想使用这些元素做其他的事情,我们只需要继承功能层次的类即可,如果我们想增加的新元素,我们只需要修改实现层次的上下两层的方法,然后就可以使用了,这样思路非常清晰,将功能和组成功能的子功能的实现隔离开来,可以随意的组合,便于组件化编程,比如我们将实现层次作为组件,我们只需要使用委托(组合)将我们想要实现的功能托付给实现层次来完成就好了,大大的提高了可重用性。那么到底什么是桥接呢,在哪里体现的?我想大家都知道了,那就是在委托的地方(private DisplayImpl displayImpl;)体现了,这就是桥,一座沟通功能层次和实现层次的桥。由此我们也看到,设计模式其实就是为了最大限度的实现代码的可重用性,为此可谓是绞尽脑汁,实现了之后就能够组件化,可移植,可扩展,从而高内聚低耦合,设计模式,我们已经渐渐地感受到了提取的初衷和意义。



推荐阅读
  • 深入解析Java虚拟机(JVM)架构与原理
    本文旨在为读者提供对Java虚拟机(JVM)的全面理解,涵盖其主要组成部分、工作原理及其在不同平台上的实现。通过详细探讨JVM的结构和内部机制,帮助开发者更好地掌握Java编程的核心技术。 ... [详细]
  • 深入解析Spring启动过程
    本文详细介绍了Spring框架的启动流程,帮助开发者理解其内部机制。通过具体示例和代码片段,解释了Bean定义、工厂类、读取器以及条件评估等关键概念,使读者能够更全面地掌握Spring的初始化过程。 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 本文深入探讨了SQL数据库中常见的面试问题,包括如何获取自增字段的当前值、防止SQL注入的方法、游标的作用与使用、索引的形式及其优缺点,以及事务和存储过程的概念。通过详细的解答和示例,帮助读者更好地理解和应对这些技术问题。 ... [详细]
  • 深入理解Java字符串池机制
    本文详细解析了Java中的字符串池(String Pool)机制,探讨其工作原理、实现方式及其对性能的影响。通过具体的代码示例和分析,帮助读者更好地理解和应用这一重要特性。 ... [详细]
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 实用正则表达式有哪些
    小编给大家分享一下实用正则表达式有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下 ... [详细]
  • 主板IO用W83627THG,用VC如何取得CPU温度,系统温度,CPU风扇转速,VBat的电压. ... [详细]
  • 本文探讨了如何通过预处理器开关选择不同的类实现,并解决在特定情况下遇到的链接器错误。 ... [详细]
  • 本文介绍了如何在 Node.js 中使用 `setDefaultEncoding` 方法为可写流设置默认编码,并提供了详细的语法说明和示例代码。 ... [详细]
  • 历经三十年的开发,Mathematica 已成为技术计算领域的标杆,为全球的技术创新者、教育工作者、学生及其他用户提供了一个领先的计算平台。最新版本 Mathematica 12.3.1 增加了多项核心语言、数学计算、可视化和图形处理的新功能。 ... [详细]
  • 深入解析SpringMVC核心组件:DispatcherServlet的工作原理
    本文详细探讨了SpringMVC的核心组件——DispatcherServlet的运作机制,旨在帮助有一定Java和Spring基础的开发人员理解HTTP请求是如何被映射到Controller并执行的。文章将解答以下问题:1. HTTP请求如何映射到Controller;2. Controller是如何被执行的。 ... [详细]
  • 本文详细介绍了装饰者(Decorator)模式,这是一种动态地为对象添加职责的方法。与传统的继承方式不同,装饰者模式通过组合而非继承来实现功能扩展,从而提供更大的灵活性和可维护性。 ... [详细]
  • Linux环境下进程间通信:深入解析信号机制
    本文详细探讨了Linux系统中信号的生命周期,从信号生成到处理函数执行完毕的全过程,并介绍了信号编程中的注意事项和常见应用实例。通过分析信号在进程中的注册、注销及处理过程,帮助读者理解如何高效利用信号进行进程间通信。 ... [详细]
author-avatar
彩色蜗牛
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有