美文网首页demoiOS之设计模式iOS组件化
设计模式系列11--桥接模式

设计模式系列11--桥接模式

作者: 西木柚子 | 来源:发表于2016-12-19 08:24 被阅读464次
    image

    假设要实现一个给客户发送提示消息的功能,发送的消息类型可分为:普通消息、加急消息、特加急消息等等,而每种消息的发送的方式一般有:系统内推送、手机短信、电子邮件等等。如果让我们来实现,会怎么做呢?

    我们先来实现一个简单的版本,使用系统推送和电子邮件发送普通消息,实现起来不叫简单,就不展示代码了,直接看UML结构图

    image

    很简单的实现对吧,现在再增加一个加急消息的发送,也是通过系统推送和Email两种方式发送,而且加急消息还额外多了一个方法,想了想简单嘛,直接扩展现有的接口不就可以了吗,此时UML结构如如下:

    image

    如果在增加一个特加急消息的发送呢,也是两种方式,也有一个额外的方法,继续扩展嘛,此时UML机构如如下:

    image

    是不是感觉类的数目剧增了,还没完呢。现在需要给每种消息类型增加一种发送方式:手机发送。继续改,UML结构图如下:

    image

    此时类的数目已经非常多了,如果继续增加消息类型或者发送方式,那么又要重复扩展,类的数目会急剧增加。

    我们来仔细分析下上面的实现,其实这里面有两个变化的维度:消息类型和发送方式。他们之间是交织在一起的,如下图所示:

    image

    由于这两个维度是交织在一起,那么他们之间的组合方式就有9种,也就是我们上面看到的有9个类,如果此时任何一个维度发生变化,都会导致与之关联的另一个维度也需要修改。而且一个维度的上的数目的增加,都会导致整体类的数目成倍数的增加。如果消息的发送类型和发送方式都有10种的话,那么就需要实现100个类,想想就恐怖。

    那么自然而然的我们就想到把这两个维度分开,让他们独自变化,互不影响,需要用的是会把他们组合在一起就行了。这样一个维度的类增加,不会导致另一个维度类也需要跟着一起增加。

    再次证明:多用组合少用继承

    那么如何实现呢?这就要请出我们今天的主角:桥接模式。


    定义

    将抽象部分与它的实现部分分离,使它们都可以独立地变化。

    抽象部分和实现部分就是两个不同的维度,抽象部分对应上面的消息类型,实现部分对应上面的消息发送方式。现在我们独立实现两个部分,然后消息类型部分想调用消息实现部分去发送消息该怎么办呢?很简单嘛,让抽象部分持有实现部分的接口,面向接口编程就可以了,这就是桥接模式名字的由来。桥接抽象部分和实现部分,下面看UML结构图会更加的清晰。


    UML结构如及说明

    image

    可以看到抽象部分的抽象类和实现部分的接口是聚合关系,表示抽象部分持有实现部门的接口,这样抽象部分就可以调用实现部分完成功能了。

    分析到这里,大家应该对桥接模式有一个大致的了解了吧,下面就来看看如何使用桥接模式来实现上面的消息发送功能。


    代码实现

    1、消息类型抽象类

    #import <Foundation/Foundation.h>
    #import "messageImplement.h"
    
    @interface abstractMessage : NSObject
    @property(strong,nonatomic)id<messageImplement> messageIm;
    
    -(void)send:(NSMutableString*)message;
    - (instancetype)initWithImplement:(id<messageImplement>)implement;
    @end
    
    ====================
    #import "abstractMessage.h"
    
    @implementation abstractMessage
    
    - (instancetype)initWithImplement:(id<messageImplement>)implement
    {
        self = [super init];
        if (self) {
            self.messageIm = implement;
        }
        return self;
    }
    
    -(void)send:(NSMutableString*)message{
        
    }
    
    @end
    
    

    2、具体消息类型

    下面只展示了普通消息的具体类实现,其他两种方式类似,详细见demo

    #import "abstractMessage.h"
    
    @interface commonMessage : abstractMessage
    
    @end
    
    
    ====================
    
    #import "commonMessage.h"
    
    @implementation commonMessage
    
    -(void)send:(NSMutableString *)message{
        [message insertString:@"【普通消息:" atIndex:0];
        [message appendString:@"】"];
        [self.messageIm sendMessage:message];
    }
    @end
    
    

    3、消息发送接口

    #import <Foundation/Foundation.h>
    
    @protocol messageImplement <NSObject>
    
    -(void)sendMessage:(NSString *)message;
    
    @end
    
    

    4、具体的消息发送方式

    下面只展示使用系统内推送方式发送消息的方式,其他两种消息发送方式类似,不在展示,具体见demo

    #import <Foundation/Foundation.h>
    #import "messageImplement.h"
    
    @interface messageSMS : NSObject<messageImplement>
    
    @end
    
    
    =================
    
    #import "messageSMS.h"
    
    @implementation messageSMS
    
    -(void)sendMessage:(NSString *)message{
        NSLog(@"使用系统内消息方式发送消息,消息内容:%@", message);
    }
    @end
    
    

    5、测试

     id<messageImplement> messageIMP = [messageMobile new];
     abstractMessage *message = [[specialUrgencyMessage alloc]initWithImplement:messageIMP];
     NSMutableString *mStr = [[NSMutableString alloc]initWithString:@"大海啊,全是水,骏马啊,四条腿"];
     [message send:mStr];
    
    

    6、输出

    2016-12-15 16:56:43.356 桥接模式[66573:2541266] 使用手机方式发送消息,消息内容:【特别加急消息:大海啊,全是水,骏马啊,四条腿】
    
    

    你可以任意组合消息类型和消息发送方式,此时类的数目只有6个,比之前的继承实现方式9个少了。如果两者都有10种实现方式,那么使用继承方式就需要100个类,而使用桥接模式只要20个类,看到了桥接模式的巨大优点吧。


    优缺点

    image

    思考

    桥接模式,需要理解桥接二字的由来,看上面的UML图就明白了,是在抽象和实现之间桥接。因为他们现在分开了,但是抽象部分必须使用实现部分去实现功能,所以抽象部分必须引用实现部分,这就是桥接。

    桥接模式对比继承的有点是把本来混在一起的两个变化未读分开,让他们独立变化,这样互相不影响,减少了类的数目,也方便扩展,而且可以动态替换功能。比如上面的消息发送功能,同样是发送普通消息,我可以选择手机、email其中的任何一种方式,只需要组合起来就行了,比继承更加灵活。

    扩展下去,我们上面只是两个维度在变化,那么如果是三个、四个维度在变化呢?如果你使用继承去实现,那就完了,不知道要重复写多少代码,而使用桥接模式,就可以把这些变化维度全部独立分开实现,然后客户端想怎么组合就怎么组合。


    Demo下载

    桥接模式Demo

    相关文章

      网友评论

        本文标题:设计模式系列11--桥接模式

        本文链接:https://www.haomeiwen.com/subject/tjozmttx.html