现有一个需求,一个游戏系统需要构建不同风格的房屋,暂不考虑其他设计模式,需要能实现在PC端、移动端....等等多个平台的构建.最简单的实现方式如下:

        /// <summary>
/// 房屋抽象
/// </summary>
public abstract class House
{
/// <summary>
/// 不同平台的初始化方法
/// </summary>
public abstract void PlatformInit(); /// <summary>
/// 绘制窗户
/// </summary>
public abstract void DrowWindow(); /// <summary>
/// 绘制门
/// </summary>
public abstract void DrowDoor();
} /// <summary>
/// PC版的现代化房屋构建过程
/// </summary>
public class PCModernHouse : House
{ public override void DrowDoor()
{ } public override void DrowWindow()
{ } public override void PlatformInit()
{
//这里初始化构建PC版的现代化房屋所需要的PC平台的相关信息
}
} /// <summary>
/// 移动端的现代化房屋构建过程
/// </summary>
public class MobileModernHouse : House
{
public override void DrowDoor()
{ } public override void DrowWindow()
{ } public override void PlatformInit()
{
//这里初始化构建移动端的现代化房屋所需要的移动端平台的相关信息
}
} /// <summary>
/// PC版经典房屋
/// </summary>
public class PCClassicalHouse : House
{
public override void DrowDoor()
{ } public override void DrowWindow()
{ } public override void PlatformInit()
{
//这里初始化构建PC版经典房屋所需要的PC平台的相关信息
}
} /// <summary>
/// 移动版经典房屋
/// </summary>
public class MobileClassicalHouse : House
{
public override void DrowDoor()
{ } public override void DrowWindow()
{ } public override void PlatformInit()
{
//这里初始化构建移动版经典房屋所需要的移动版平台的相关信息
}
}

ok,这种设计虽然能很好的满足需求,但是这里只列举了两种平台,如果之后需要支持更多的平台,这种设计模式显然不行,因为产生的类会原来越多.随之维护的成本也越来越大.

so.分析代码,发现设计中有两个变化点,一个是House,另一个是平台,而上面的代码并没有把平台作为变化点抽离出来,所以需要将平台抽离抽来,接下来,对代码进行重构.

思考:由于House的固有逻辑,使得House具有两个维护的变化一一一个变化维度为"平台的变化",另一个变化维度为"风格的变化".那么如何解决这种维度的变化?如何利用OOP,来轻松地使得House以"平台"和"风格"两个方向变化?这就是桥接模式要解决的问题:

利用Bridge桥接模式重构后的代码如下:

        #region 抽象
/// <summary>
/// 房屋抽象
/// </summary>
public abstract class House
{
protected HousePlatform _housePlatform; public House(HousePlatform housePlatform)
{
_housePlatform = housePlatform;
} public HousePlatform HousePlatform => _housePlatform; /// <summary>
/// 绘制窗户
/// </summary>
public abstract void DrowWindow(); /// <summary>
/// 绘制门
/// </summary>
public abstract void DrowDoor();
} /// <summary>
/// 平台抽象
/// </summary>
public abstract class HousePlatform
{
public abstract HousePlatform Init();
}
#endregion #region 实现 /// <summary>
/// 现代化房屋
/// </summary>
public class ModernHouse : House
{
private HousePlatform _platform;
public ModernHouse(HousePlatform housePlatform) : base(housePlatform)
{
_platform = HousePlatform.Init();
} public override void DrowDoor()
{
//执行HousePlatform实例的一些方法,然后做自己的逻辑
} public override void DrowWindow() { }
} /// <summary>
/// 经典房屋
/// </summary>
public class ClassicalHouse : House
{
private HousePlatform _platform;
public ClassicalHouse(HousePlatform housePlatform) : base(housePlatform)
{
_platform = housePlatform.Init();
} public override void DrowDoor() { } public override void DrowWindow() { }
} /// <summary>
/// PC平台
/// </summary>
public class PCPlatform : HousePlatform
{
public override HousePlatform Init()
{
//执行PC平台的初始化方法,返回一个PCPlatform对象实例供对应的房屋使用
return this.MemberwiseClone() as HousePlatform;
}
} /// <summary>
/// 移动端平台
/// </summary>
public class MobilePlatform : HousePlatform
{
public override HousePlatform Init()
{
//执行移动端平台的初始化方法,返回一个MobilePlatform对象实例供对应的房屋使用
return this.MemberwiseClone() as HousePlatform;
}
} #endregion

客户端调用代码如下:

        public class GameSystem
{
public void Build()
{
//构建了一个基于PC平台的现代化房屋
var modernHouse = new ModernHouse(new PCPlatform());
modernHouse.DrowDoor();
modernHouse.DrowWindow();
}
}

将变化点平台也分离出来,并抽象出一个接口,并通过构造函数的方式将两个变化点桥接到一起,可以理解为组合到一起,这就是桥接模式.

桥接模式使用的要点:

1、Bridge桥接模式一般是用在两个变化剧烈的维度,如上面的房屋的风格和平台的类型都是变化非常剧烈的,但如果有一个维度变化并不剧烈,则并需要使用桥接模式.

2、抽象和实现沿着各自纬度的变化,比如不同风格的房屋和不同的平台.得到各个子类之后,便可以不同的方式组合它们,便可以得到不同平台下的不同风格的房屋.

3、多继承关系,可以考虑使用Bridge模式来实现.多继承方案往往违背oop的单一职责.Bridge模式是比多继承更好的选择.

4、Bridge模式使用对象组合关系,解耦了抽象和实现之间固有的绑定关系,使得抽象(不同风格的房屋)和实现(不同的平台)可以沿着各自的维度变化.

,

Bridge桥接模式(结构型模式)的更多相关文章

  1. 代理模式/proxy模式/结构型模式

    代理模式proxy 定义 为其他对象提供一种代理,并以控制对这个对象的访问.最简单的理解,买东西都是要去商店的,不会去工厂. java实现三要素 proxy(代理)+subject(接口)+realS ...

  2. 设计模式(十二): Flyweight享元模式 -- 结构型模式

    说明: 相对于其它模式,Flyweight模式在PHP实现似乎没有太大的意义,因为PHP的生命周期就在一个请求,请求执行完了,php占用的资源都被释放.我们只是为了学习而简单做了介绍. 1. 概述 面 ...

  3. 设计模式(十三): Proxy代理模式 -- 结构型模式

      设计模式(十一)代理模式Proxy(结构型) 1.概述 因为某个对象消耗太多资源,而且你的代码并不是每个逻辑路径都需要此对象, 你曾有过延迟创建对象的想法吗 ( if和else就是不同的两条逻辑路 ...

  4. 设计模式学习之路——Facade 外观模式(结构型模式)

    动机: 组件的客户和组件中各种复杂的子系统有了过多的耦合,随着外部客户程序和各子系统的演化,这种过多的耦合面临很多变化的挑战.如何简化外部客户程序和系统间的交互接口?如何将外部客户程序的演化和内部子系 ...

  5. 设计模式(十一):FACADE外观模式 -- 结构型模式

    1. 概述 外观模式,我们通过外观的包装,使应用程序只能看到外观对象,而不会看到具体的细节对象,这样无疑会降低应用程序的复杂度,并且提高了程序的可维护性.例子1:一个电源总开关可以控制四盏灯.一个风扇 ...

  6. 设计模式(十):Decorator装饰者模式 -- 结构型模式

    1. 概述 若你从事过面向对象开发,实现给一个类或对象增加行为,使用继承机制,这是所有面向对象语言的一个基本特性.如果已经存在的一个类缺少某些方法,或者须要给方法添加更多的功能(魅力),你也许会仅仅继 ...

  7. 设计模式(八):Bridge桥接模式 -- 结构型模式

    1. 概述 在软件系统中,某些类型由于自身的逻辑,它具有两个或多个维度的变化,那么如何应对这种“多维度的变化”?如何利用面向对象的技术来使得该类型能够轻松的沿着多个方向进行变化,而又不引入额外的复杂度 ...

  8. 适配器模式/adapter模式/结构型模式

    定义 将类的接口转化为客户端希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的类可以一起工作,别名Wrapper(包装器). 适配器模式,最终改变一个已有对象的接口. 使用场景 当有那么个类, ...

  9. 外观模式/facade模式/结构型模式

    外观模式 为子系统中的一组接口提供一个一致的界面, Facade模式定义了一个高层接口,这个接口使得这一子系统更加容易使用. 外观模式三要素(client-facade-subSystem) 外观角色 ...

随机推荐

  1. adb 调试出现问题

    用adb shell时出现error: insufficient permissions for device 在终端输入 sudo -s adb kill-server adb start-serv ...

  2. [指南] 15分钟学会MySQL(Linux版)

    原文链接:http://www.mysqlpub.com/thread-348-1-1.html 原创出处:MySQLpub.com  , 作者:kider  ,转载请注明作者和出处,并不能用于商业用 ...

  3. 强大的DataGrid组件[1]

    说明:DataGrid组件是Silverlight数据组件中最为常用并且是功能最为强大的数据组件.因此,对开发者而言,深入了解其特性是十分有必要的.本文先介绍该组件的基本特性,接着通过几个简单实例来说 ...

  4. 18) maven 项目结构:继承

    Project Inheritance [ɪn'herɪt(ə)ns] https://maven.apache.org/guides/introduction/introduction-to-the ...

  5. (转)什么是.NET?什么是CLI?什么是CLR?IL是什么?JIT是什么,它是如何工作的?GC是什么,简述一下GC的工作方式?

    转自:http://www.cnblogs.com/haofaner/articles/2288968.html 1:什么是.NET? NET 是 Microsoft 的用以创建 XML Web 服务 ...

  6. (01背包)Buy the souvenirs (hdu 2126)

    http://acm.hdu.edu.cn/showproblem.php?pid=2126 Buy the souvenirs Time Limit: 10000/1000 MS (Java/Oth ...

  7. 04:第一个OC类

    1.类与对象的关系 面向对象的核心就是对象,那怎么创建对象? OC中创建对象比较复杂, 首先要理解一个概念叫做类. 现实生活中是根据一份描述,一份模板创建对象,编程语言也一样,也必须先有一份描述,在这 ...

  8. poj3929

    题意: 如上图放置的一个圆锥,告诉你从圆锥顶的洞中流出多少体积的水,求现在水面高度.. 思路: 无聊时做的一道题,实际上就是一道高数题,重积分,可惜我高数本来也不好而且还忘光了,积了很久,而且错了很多 ...

  9. SRM476

    250pt 题意:饲养N<=50只Badgers,每只食量是X[i],当没看到别的一只Badgers吃东西时,它的食量就会增加Y[i],现在一共用P的粮食,问最多能养的起多少只獾. 思路:枚举一 ...

  10. delphi 分享三个随机字符串

    uses math; function GenID:String; var b, x: byte; begin Result := '{'; Randomize; do begin ) > ,) ...