博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
iOS开发-适配器和外观模式
阅读量:6715 次
发布时间:2019-06-25

本文共 2638 字,大约阅读时间需要 8 分钟。

适配器模式,属于结构型模式,其主要作用是将一个类的接口转换成客户希望的另外一个接口。适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。适配器模式有对象适配器和类适配器两种,类适配器模式需要语言实现多继承,OC不支持多继承,所以一般我们都实现对象适配器。外观模式提供了一个统一的接口,用来访问子系统中的一群接口,外观定义了一个高层接口,让子系统更容易使用。适配器是为了转换接口,外观模式是为了简化接口。

适配器模式

对象适配器模式UML类图:

关于适配模式最常见的就是手机充电的例子,通过数据线和转换口连接,然后将转换口和插座连接,转换口就可以理解为适配器~

数据线(Target):

1
2
3
4
5
6
7
8
9
10
11
@protocol DataLineProtocol <NSObject>
 
@optional
-(
void
)connect;
 
@end
 
@
interface 
DataLine : NSObject<DataLineProtocol>
 
 
@end

电源插头(Adapter):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
@
interface 
DataAdapter(){
    
EleSocket  *eleSocket;
}
@end
 
@implementation DataAdapter
 
-(instancetype)initWithEleSocket:(EleSocket *)socket{
    
self=[super init];
    
if 
(self) {
        
eleSocket=socket;
    
}
    
return 
self;
}
 
-(
void
)connect{
    
[eleSocket speicalConnect];
}
 
@end

被适配者(插座):

1
2
3
4
5
6
7
@implementation EleSocket
 
-(
void
)speicalConnect{
    
NSLog(
@"specialConnect--电源接通"
);
}
 
@end

客户端(手机):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@
interface 
Mobile(){
    
DataLine *dataLine;
}
 
@end
 
@implementation Mobile
 
-(instancetype)initWithDataLine:(DataLine *)line{
    
self=[super init];
    
if 
(self) {
        
dataLine=line;
    
}
    
return 
self;
}
 
-(
void
)charge{
    
[dataLine connect];
}
 
@end

客户端调用:

1
2
3
4
EleSocket  *socket=[[EleSocket alloc]init];
DataLine   *line=[[DataAdapter alloc] initWithEleSocket:socket];
Mobile     *moblie=[[Mobile alloc]initWithDataLine:line];
[moblie charge];

测试结果:

 

外观模式

外观模式通过引入一个外观角色来简化客户端与子系统之间的交互,为复杂的子系统调用提供一个统一的入口,降低子系统与客户端的耦合度,且客户端调用非常方便。适配器是是接口转换,外观是简化接口,将多个子系统接口转换为一个,最简单的例子就是在饭店吃饭和在家吃饭,如果去饭店吃饭,只需要下单,等个十几二十分钟我们就可以吃饭,在家自己做饭就需要买菜,淘米,炒菜等等一系列动作。

UML类图:

 

我们简单的模拟一下餐馆点餐的过程需要饭店,服务员,厨师:

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
@
interface 
Hotel(){
    
Waiter  *myWaiter;
    
Cook    *myCook;
}
 
@end
 
@implementation Hotel
 
-(instancetype)initWithCook:(Waiter *)waiter cook:(Cook *)cook{
    
self=[super init];
    
if 
(self) {
        
myWaiter=waiter;
        
myCook=cook;
    
}
    
return 
self;
}
-(
void
)order{
    
[myWaiter handleOrder];
    
[myCook cook];
    
[myCook complete];
}
 
@end

 

客户端调用:

1
2
Hotel  *hotel=[[Hotel alloc]initWithCook:[Waiter 
new
] cook:[Cook 
new
]];
[hotel order];

  

 

关于门面模式的简单实现中我们通过一个接口,可以调用子系统的接口,非常简单,外观模式的优势有以下几点:

①对客户屏蔽子系统组件,减少了客户处理的对象数目并使得子系统使用起来更加容易。通过引入外观模式,客户代码将变得很简单,与之关联的对象也很少。
②实现了子系统与客户之间的松耦合关系,这使得子系统的组件变化不会影响到调用它的客户类,只需要调整外观类即可。
降低了大型软件系统中的编译依赖性,并简化了系统在不同平台之间的移植过程,因为编译一个子系统一般不需要编译所有其他的子系统。一个子系统的修改对其他子系统没有任何影响,而且子系统内部变化也不会影响到外观对象。
④提供了一个访问子系统的统一入口,并不影响用户直接使用子系统类。
外观模式的缺点:
不能很好地限制客户使用子系统类,如果对客户访问子系统类做太多的限制则减少了可变性和灵活性。
② 在不引入抽象外观类的情况下,增加新的子系统可能需要修改外观类或客户端的源代码,违背了“开闭原则”。

 本文的子系统对象可以从外部传递,如果不希望从外部传递,可以在初始化的时候内部对子系统的进行初始化~

本文转自Fly_Elephant博客园博客,原文链接:http://www.cnblogs.com/xiaofeixiang/p/5123168.html,如需转载请自行联系原作者

你可能感兴趣的文章
SequoiaDB 笔记
查看>>
lduan HyPer-V 网络存储(三)
查看>>
SSH 命令行参数详解【英】
查看>>
前端技术学习之选择器(四)
查看>>
2016年4月4日中项作业
查看>>
log4j配置
查看>>
centos备份与还原
查看>>
fixed 兼容ie6
查看>>
条件+努力=?
查看>>
hadoop常用服务管理命令
查看>>
洛谷P4169 天使玩偶 (算竞进阶习题)
查看>>
Order By操作
查看>>
(三)mybatis之对Hibernate初了解
查看>>
nginx安装与配置
查看>>
Android 命令设置获取、IP地址、网关、dns
查看>>
查找当前薪水(to_date='9999-01-01')排名第二多的员工编号emp_no、薪水salary、last_name以及first_name,不准使用order by...
查看>>
[SQL in Azure] Windows Azure Virtual Machine Readiness and Capacity Assessment
查看>>
关于CCR测评器的自定义校验器(Special Judge)
查看>>
java设计模式之 装饰器模式
查看>>
利息力(force of interest)
查看>>