java自学网VIP

Java自学网

 找回密码
 立即注册

QQ登录

只需一步,快速开始

查看: 19075|回复: 105

Java面向对象设计之桥接模式

  [复制链接]
  • TA的每日心情
    开心
    2021-5-25 00:00
  • 签到天数: 1917 天

    [LV.Master]出神入化

    2090

    主题

    3748

    帖子

    6万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    66756

    宣传达人突出贡献优秀版主荣誉管理论坛元老

    发表于 2019-11-1 22:19:36 | 显示全部楼层 |阅读模式
    模式动机
    设想如果要绘制矩形、圆形、椭圆、正方形,我们至少需要4个形状类,但是如果绘制的图形需要具有不同的颜色,如红色、绿色、蓝色等,此时至少有如下两种设计方案:
    • 第一种设计方案是为每一种形状都提供一套各种颜色的版本。
    • 第二种设计方案是根据实际需要对形状和颜色进行组合
      4 M* J2 V+ k0 w$ r7 X. D
    对于有两个变化维度(即两个变化的原因)的系统,采用方案二来进行设计系统中类的个数更少,且系统扩展更为方便。设计方案二即是桥接模式的应用。桥接模式将继承关系转换为关联关系,从而降低了类与类之间的耦合,减少了代码编写量。
    模式定义
    桥接模式(Bridge Pattern):将抽象和实现解耦,使得两者可以独立地变化。它是一种对象结构型模式,又称为桥梁模式。
    模式角色
    桥接模式包含如下角色:
    • Abstraction:抽象类角色
    • RefinedAbstraction:扩充抽象类
    • Implementor:实现化角色
    • ConcreteImplementor:具体实现化角色的实现类
      . O; g- j/ c. Q6 B9 r( S" t, |
    UML类图7 h/ L$ V6 e' N- M  m, J
    桥接模式UML
    0 W' I' k" c! T7 p
    ; J* W. s9 V% w  E+ |
    2 o8 ?7 g1 E! I  @1 y9 d
    代码示例
    首先,是实现化角色Abstraction类:
    1. /**
      : X7 z% g$ i+ q9 h/ d$ e
    2. * 实现化角色 Implementor.
      : j4 [; |) U  I7 W% ?6 x% n
    3. *
      3 r! @8 X+ x# t! c
    4. * @author blinkfox on 2018-12-17.
      9 U7 Y' A8 }+ z- T
    5. */; G5 v- R4 W1 J5 w; t, v2 X
    6. public interface Implementor {2 E& `, n- w  k# n2 s3 F

    7. 8 R  G2 @! G1 Y! G5 m7 @2 s; n
    8.     /**( d: y4 {: x" h/ j" s
    9.      * 基本方法1.
      1 x' ?) ^1 _3 ]" t# e
    10.      */
      ) A& Z: E/ v3 X% N" |; B# `: H
    11.     void doSomething();" J8 t- b+ [9 i/ ?1 e# o) o: T
    12. 4 W! p. c1 g" t% J, X; Z5 s* y# X# t
    13.     /**$ v  X! }# a6 v9 D, ~; U
    14.      * 基本方法2.& s. S+ z9 @3 W% p8 t
    15.      */2 i0 Q* {- ?2 ~) W5 d( Y
    16.     void doAnything();
      2 `7 B% W& c" [  e: y0 O

    17. / t' p" H; q. ]4 Z' X0 p
    18. }
    复制代码

    & u2 F- Y; p: X! @7 h) B
    然后,是各个具体的实现化角色类:
    1. /**
      + Y* u( f) I% b& z! P
    2. * ConcreteImplementor1., B) K; N2 n2 K1 |( v+ O8 X4 n) l
    3. *
      # P8 A- r, B, W
    4. * @author blinkfox on 2018-12-17.
      : |& t( q2 B6 {7 ~+ o0 C
    5. */" L  {# L  ?% G  J# D
    6. public class ConcreteImplementor1 implements Implementor {
      . D* m/ S* K# k9 m
    7. 6 e' A. u) x7 f! h' s
    8.     /**1 W: F+ m  p0 @" D7 p
    9.      * 基本方法1.5 ?2 N1 P, n# y8 e+ y
    10.      */: b+ _( @' k  H" D$ V
    11.     @Override
      # a; b* S/ Y; c
    12.     public void doSomething() {' h9 i$ K- U7 P2 Z& v( u& S
    13.         System.out.println("ConcreteImplementor1 的业务逻辑 doSomething.");4 v5 @# s. Q5 E3 [+ Y) q0 A
    14.     }% n9 C  A! L0 |% e. A5 P; ]. I6 A" J
    15. ) K/ Z- v8 F8 J9 Y6 O# h9 g
    16.     /**$ [( N! l) s$ }1 q
    17.      * 基本方法2.' s% R3 K) N9 n( I
    18.      */# e! ~2 j. J& Q0 J
    19.     @Override
      $ W( g7 z. T: |
    20.     public void doAnything() {
      + h# A) f/ A- _0 b
    21.         System.out.println("ConcreteImplementor1 的业务逻辑 doAnything.");
      $ ~( N5 n% @; \7 Z# u3 J
    22.     }
      : u: \2 s) {! r& k5 Q9 `/ J

    23. 2 E5 A' o' Y  q5 y8 t9 c
    24. }
    复制代码
    1. /**
      3 V( p! C+ R. h# ]6 ~1 w
    2. * ConcreteImplementor2.# s- S: k5 `; V9 k! h4 p
    3. *' `* i& r% r8 X! o
    4. * @author blinkfox on 2018-12-17.& Q3 _5 _/ e' ]9 m8 D
    5. */
      0 }0 N; ?2 P, y. |: {1 }
    6. public class ConcreteImplementor2 implements Implementor {1 m2 ^* Z9 n7 L; n4 w" N" a5 [' b! d
    7. 6 c( \2 C( x7 o0 Z+ V" _8 _6 |5 |
    8.     /**
      ; x" y1 s; ~/ i4 k- Z' ^: ~
    9.      * 基本方法1.
      6 d3 Z8 o6 K. A( R* J% ?9 Y2 g/ ?$ U
    10.      */
      % I  _* r0 o. r+ M* g8 p( W
    11.     @Override
      1 i3 b) S6 t6 i( Y
    12.     public void doSomething() {
      - g3 x# `+ u$ L5 \" w
    13.         System.out.println("ConcreteImplementor2 的业务逻辑 doSomething.");8 A3 L, ]. D. W0 D" U! P& G* z: U
    14.     }* P! ]  Z. H8 q- I4 f& u7 v* E- _8 ?
    15. 2 u9 z1 b5 m1 F
    16.     /**. H" i/ c) l& a5 {% M1 s) Q& v
    17.      * 基本方法2.
      6 W& r$ S5 n/ C' u. m, X
    18.      */  Q$ |. b- u  P
    19.     @Override
      3 u, l9 U8 U9 D( _4 n  Y( E+ ?
    20.     public void doAnything() {
      2 E  X4 K3 o# B& P8 z& Q8 W8 p  U
    21.         System.out.println("ConcreteImplementor2 的业务逻辑 doAnything.");
      ' @0 Y! |; T5 m# M3 h: g$ S/ i
    22.     }
      & Q! \- I% U2 l5 {
    23. 3 ~9 t+ p9 i* _+ o& A. D, |
    24. }
    复制代码
    5 W+ b( X% `% \6 ~& D
    接下来,是抽象类角色Abstraction类:
    1. package com.blinkfox.patterns.bridge;$ Y: v2 h& n" ^* `& B
    2. . X) L& _/ b9 k8 u
    3. /**# O, `  h: a. ]* v+ X
    4. * 抽象化角色 Abstraction.
      # J* l) M; g" z. v' d
    5. *" u. F0 D- \" M+ F
    6. * @author blinkfox on 2018-12-17.
      4 p" M/ w5 v, m3 I8 Q
    7. */& Q; s$ `# p2 k7 _
    8. public abstract class Abstraction {& {6 K/ u, W8 N9 r" o3 E

    9. + o6 G0 b) b! Q* b
    10.     /** 定义对实现化角色的引用. */, G7 w" n- M. q$ L* j( h
    11.     private Implementor impl;* {" s+ r, j; X
    12. , R8 w) f# i! Z' Y8 o1 g1 }  d/ R  U
    13.     /**5 ~$ |" _# O+ n: v5 L: R
    14.      * 构造方法.
      ! S' _3 Y' O+ n8 g! Z! Y+ Y
    15.      *0 s0 e, ~/ q% k4 u3 G3 v! Z* Q4 A
    16.      * @param impl 实现类的实例; j! O  U/ H: o9 S9 {  P
    17.      */
      4 Y/ q1 D2 B' A+ a+ ^+ r
    18.     public Abstraction(Implementor impl) {
      / ~' e4 g" k7 ]  j$ f/ W! \
    19.         this.impl = impl;
      ; x0 _! _2 ]' D# T2 [
    20.     }/ d" Y* g3 N% Z  h8 D% m$ ]
    21. + b) f- x4 S% _& P! l6 u
    22.     /**
      % @! l4 `% s5 ?# t" ?
    23.      * impl 的 getter方法.. `; \; w% U9 r: J* `% z
    24.      *8 k8 [" O( _, z( N" l, Q
    25.      * @return impl
      ' V2 I0 ~6 {0 P/ j
    26.      */
      % B! A$ ]7 M' a$ u, ]: H: S6 a) g7 ~6 h
    27.     public Implementor getImpl() {
      - p, r# p8 v% S8 p4 Q2 F% Y
    28.         return impl;
      6 t: G1 @; Z  X' D9 W# y% E& s2 M
    29.     }5 E* g! k8 ?4 L; W0 s

    30. . g# k' d+ |+ Y. q
    31.     /**
      ; c7 R2 P: ^% g, G  |- L& P" x
    32.      * 自身的请求处理方法.( \" }3 @/ K0 q8 u
    33.      */1 }  S5 ?  u& n! o+ P
    34.     public void request() {& D+ }% j. R6 d
    35.         this.impl.doSomething();
      / g+ o0 |! b# p- C
    36.     }0 E9 c( _% `) U" L' {5 E

    37. : A; j( f% V9 e/ y
    38. }
    复制代码
    / N3 v$ y- p% ^. I8 C+ v! f
    再次,是扩展的具体抽象化角色类RefinedAbstraction:
    1. /**& I6 O0 F2 `8 B) O8 K
    2. * RefinedAbstraction.
      + H3 o2 E+ d0 K- k/ q2 B9 H; N% Y
    3. *  r# X' k( g/ J) p6 [# K0 t
    4. * @author blinkfox on 2018-12-17.
      8 b8 Z, t: d8 \$ H3 _$ r
    5. */! b9 _3 P9 A5 f! g
    6. public class RefinedAbstraction extends Abstraction {& a+ N1 u) w& W. ?+ b( c1 s

    7. 0 T+ [5 v0 F. I& e
    8.     /**
      . A! P) k/ n* D3 E) o1 q0 a
    9.      * 构造方法.3 f) }& K3 O$ ^2 z6 }9 s: \
    10.      *3 [% y/ t( {6 l: f
    11.      * @param impl 实现类的实例3 q3 {9 T: j4 _2 ~( f- p, R
    12.      */
      4 ]7 P1 O7 u4 A+ I% N1 L
    13.     public RefinedAbstraction(Implementor impl) {% S0 x( o( d! [) Q
    14.         super(impl);
      * k  L( O5 U3 h5 X" E. y
    15.     }* M) }- x5 T) C3 \$ R- W- s

    16. ( z  p8 j5 Q5 x& P1 o; v  O
    17.     /**
      3 w! B+ R8 q" Y1 r! D  ~# K
    18.      * 覆盖后的请求处理方法.7 v+ D5 ~; z2 C
    19.      */' B# J3 Q$ n& d, c& [
    20.     @Override
      9 I7 s5 z$ d" v. z) o0 z1 B
    21.     public void request() {$ |, C  K+ u% W& v( y  A3 E$ ~
    22.         System.out.println("RefinedAbstraction 开始做业务处理.");* x6 b" u7 E9 {8 k, {! G  `; B
    23.         super.request();
      / K: E( V) H  F4 r
    24.         super.getImpl().doAnything();6 S) R1 Z! @, B  q
    25.     }
      * q- c$ u' T; _6 O8 \$ F/ j2 m
    26. # l9 E. B* {3 n2 Y) g, }, K! L
    27. }
    复制代码
    * W. {  z' a( d0 O6 r0 R6 m0 |
    最后,是客户端场景类:
    1. /**! v# T9 `+ \2 G( h' ]8 W: w+ }
    2. * Client.! T5 i' x+ R) d* o# z+ D9 p, \/ f  y
    3. *
      6 }" g, h6 V& d" f$ h! }* T% s
    4. * @author blinkfox on 2018-12-17.
      # E) q1 C7 p3 }- y) j
    5. */
      - Q5 a% h1 Z/ ?; L  b' k
    6. public class Client {
      9 G* S2 e8 |. j5 {. \
    7. + I( @, d6 }# g2 B5 I8 Z3 c
    8.     /**. j7 Q/ n1 J; D4 P& k/ l
    9.      * main方法.
      : H4 J  j8 ]  Y2 z" ]4 w  k1 s' Y6 ^
    10.      *
      % |* \, j& V+ H+ N1 ]$ K$ ~
    11.      * @param args 数组参数
      6 X, Q: u% d8 i$ G2 w
    12.      */
      ( ?+ ^7 r) H  C4 X
    13.     public static void main(String[] args) {/ T0 s" ]4 h2 V+ B6 _3 G/ Q) p% W
    14.         // 定义一个实现化角色和抽象化角色,并执行请求方法.
      # E9 t: [: u/ s
    15.         Implementor impl = new ConcreteImplementor1();, ]1 L, Z* ?* h! D  y0 u3 }
    16.         Abstraction abs = new RefinedAbstraction(impl);
      ; o( C7 _# `4 ?+ l5 u) }
    17.         abs.request();# {9 [$ ]+ |# M" ~
    18.     }
      ) w$ T$ z/ R2 L9 m+ }. V+ V) x
    19. $ X* w; w9 d" o2 d% \% ]
    20. }
    复制代码
    5 b! l+ X* @# R! E0 U
    模式分析
    理解桥接模式,重点需要理解如何将抽象化(Abstraction)与实现化(Implementation)脱耦,使得二者可以独立地变化。
    • 抽象化:抽象化就是忽略一些信息,把不同的实体当作同样的实体对待。在面向对象中,将对象的共同性质抽取出来形成类的过程即为抽象化的过程。
    • 实现化:针对抽象化给出的具体实现,就是实现化,抽象化与实现化是一对互逆的概念,实现化产生的对象比抽象化更具体,是对抽象化事物的进一步具体化的产物。
    • 脱耦:脱耦就是将抽象化和实现化之间的耦合解脱开,或者说是将它们之间的强关联改换成弱关联,将两个角色之间的继承关系改为关联关系。桥接模式中的所谓脱耦,就是指在一个软件系统的抽象化和实现化之间使用关联关系(组合或者聚合关系)而不是继承关系,从而使两者可以相对独立地变化,这就是桥接模式的用意。" @$ q7 d- A0 r5 d) k  f* W
    优点
    桥接模式的优点:
    • 分离抽象接口及其实现部分。
    • 桥接模式有时类似于多继承方案,但是多继承方案违背了类的单一职责原则(即一个类只有一个变化的原因),复用性比较差,而且多继承结构中类的个数非常庞大,桥接模式是比多继承方案更好的解决方法。
    • 桥接模式提高了系统的可扩充性,在两个变化维度中任意扩展一个维度,都不需要修改原有系统。
    • 实现细节对客户透明,可以对用户隐藏实现细节。. t4 S% e: Y& J" C, X+ N3 A
    缺点
    桥接模式的缺点:
    • 桥接模式的引入会增加系统的理解与设计难度,由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计与编程。
    • 桥接模式要求正确识别出系统中两个独立变化的维度,因此其使用范围具有一定的局限性。! T+ ?( a( e. j5 s
    适用环境
    在以下情况下可以使用桥接模式:
    • 如果一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系。
    • 抽象化角色和实现化角色可以以继承的方式独立扩展而互不影响,在程序运行时可以动态将一个抽象化子类的对象和一个实现化子类的对象进行组合,即系统需要对抽象化角色和实现化角色进行动态耦合。
    • 一个类存在两个独立变化的维度,且这两个维度都需要进行扩展。
    • 虽然在系统中使用继承是没有问题的,但是由于抽象化角色和具体化角色需要独立变化,设计要求需要独立管理这两者。
    • 对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用。
      ; L: w( B' @6 R5 b9 r& L0 j- g8 f: @
    模式应用
    一个Java桌面软件总是带有所在操作系统的视感(LookAndFeel),如果一个Java软件是在Unix系统上开发的,那么开发人员看到的是Motif用户界面的视感;在Windows上面使用这个系统的用户看到的是Windows用户界面的视感;而一个在Macintosh上面使用的用户看到的则是Macintosh用户界面的视感,Java语言是通过所谓的Peer架构做到这一点的。Java为AWT中的每一个GUI构件都提供了一个Peer构件,在AWT中的Peer架构就使用了桥接模式。
    模式扩展
    适配器模式与桥接模式的联用:
    桥接模式和适配器模式用于设计的不同阶段,桥接模式用于系统的初步设计,对于存在两个独立变化维度的类可以将其分为抽象化和实现化两个角色,使它们可以分别进行变化;而在初步设计完成之后,当发现系统与已有类无法协同工作时,可以采用适配器模式。但有时候在设计初期也需要考虑适配器模式,特别是那些涉及到大量第三方应用接口的情况。
    总结
    • 桥接模式将抽象部分与它的实现部分分离,使它们都可以独立地变化。它是一种对象结构型模式。
    • 桥接模式包含如下四个角色:抽象类中定义了一个实现类接口类型的对象并可以维护该对象;扩充抽象类扩充由抽象类定义的接口,它实现了在抽象类中定义的抽象业务方法,在扩充抽象类中可以调用在实现类接口中定义的业务方法;实现类接口定义了实现类的接口,实现类接口仅提供基本操作,而抽象类定义的接口可能会做更多更复杂的操作;具体实现类实现了实现类接口并且具体实现它,在不同的具体实现类中提供基本操作的不同实现,在程序运行时,具体实现类对象将替换其父类对象,提供给客户端具体的业务操作方法。
    • 在桥接模式中,抽象化(Abstraction)与实现化(Implementation)脱耦,它们可以沿着各自的维度独立变化。
    • 桥接模式的主要优点是分离抽象接口及其实现部分,是比多继承方案更好的解决方法,桥接模式还提高了系统的可扩充性,在两个变化维度中任意扩展一个维度,都不需要修改原有系统,实现细节对客户透明,可以对用户隐藏实现细节;其主要缺点是增加系统的理解与设计难度,且识别出系统中两个独立变化的维度并不是一件容易的事情。
    • 桥接模式适用情况包括:需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免在两个层次之间建立静态的继承联系;抽象化角色和实现化角色可以以继承的方式独立扩展而互不影响;一个类存在两个独立变化的维度,且这两个维度都需要进行扩展;设计要求需要独立管理抽象化角色和具体化角色;不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统。9 t' v+ Y, j, B" u
    回复

    使用道具 举报

    该用户从未签到

    0

    主题

    4588

    帖子

    9178

    积分

    普通会员

    Rank: 2

    积分
    9178
    发表于 2019-12-24 19:44:21 | 显示全部楼层
    佩服佩服!
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    0

    主题

    4555

    帖子

    9112

    积分

    普通会员

    Rank: 2

    积分
    9112
    发表于 2020-1-5 14:46:26 | 显示全部楼层
    不错 支持下
    回复 支持 反对

    使用道具 举报

  • TA的每日心情
    慵懒
    2015-7-28 14:19
  • 签到天数: 1 天

    [LV.1]初学乍练

    0

    主题

    4663

    帖子

    9344

    积分

    普通会员

    Rank: 2

    积分
    9344
    QQ
    发表于 2020-1-21 10:12:11 | 显示全部楼层
    努力学习努力学习努力学习
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    0

    主题

    4560

    帖子

    9122

    积分

    禁止发言

    积分
    9122
    发表于 2020-1-30 14:04:36 | 显示全部楼层
    提示: 作者被禁止或删除 内容自动屏蔽
    回复 支持 反对

    使用道具 举报

  • TA的每日心情
    奋斗
    2015-7-5 17:05
  • 签到天数: 1 天

    [LV.1]初学乍练

    0

    主题

    4740

    帖子

    9502

    积分

    普通会员

    Rank: 2

    积分
    9502
    发表于 2020-2-3 08:01:30 | 显示全部楼层
    非常好,顶一下
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    0

    主题

    4674

    帖子

    9350

    积分

    普通会员

    Rank: 2

    积分
    9350
    发表于 2020-2-6 09:10:23 | 显示全部楼层
    哈哈哈哈哈,开心
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    0

    主题

    4469

    帖子

    8940

    积分

    普通会员

    Rank: 2

    积分
    8940
    发表于 2020-2-22 18:21:59 | 显示全部楼层
    好资源
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    0

    主题

    4590

    帖子

    9182

    积分

    普通会员

    Rank: 2

    积分
    9182
    发表于 2020-3-1 11:26:55 | 显示全部楼层
    好资源
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    0

    主题

    4502

    帖子

    9006

    积分

    普通会员

    Rank: 2

    积分
    9006
    发表于 2020-3-1 14:23:23 | 显示全部楼层
    即将学习之路
    回复 支持 反对

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|Archiver|手机版|小黑屋|Java自学网

    GMT+8, 2025-3-26 08:27 , Processed in 0.577788 second(s), 35 queries .

    Powered by Javazx

    Copyright © 2012-2022, Javazx Cloud.

    快速回复 返回顶部 返回列表