用来解决接口适配问题的三种模式:适配器模式,桥接模式,外观模式。

一,概念

  适配器模式,将一个类的结构转换成用户希望的另一个接口,使得原本接口不兼容的类能在一起工作。换句话说,适配器模式就是链接两种不同种类的对象,使其很好的协同工作。(说的很美吧,我抄的,百度里每个帖子都会这样说。简单说就是你笔记本是19v供电,但是家里面电压是220v,这时候想要充电,就需要一个笔记本的220v转19v电源适配器)。

  有两种实现结构(类的适配器,对象的适配器),还有说三种(接口的适配器方式),额。

二,模式结构图

  了解三个角色,结合结构图理解:

    Target:目标角色定义客户端具体使用的接口,也就是我们的期望接口,简单一点就是协议,协议定义我们需要的接口

    Adaptee:源角色,你想把“谁”转换成目标角色,这个“谁”就是源角色,它是已经存在的、运行良好的类或对象。

    Adapter:适配器角色,适配器模式的核心角色,其他两个角色都是已经存在的角色,而适配器角色是需要新建立的,他的职责非常简单:把源角色转换为目标角色。

  类的适配器模式:

  

  Adapter是一个Target(目标接口)类型,同时也是一个Adaptee(被适配的类)类型。Adapter重载Target的request方法,但没有重载Adaptee的specificRequest方法,而是在其request方法中调用父类的specificRequest方法(即:[super specificRequest])。当request方法在运行时,向Adaptee发送super消息时,Adaptee按自己的方式执行specificRequest方法。注意:只有当Target是协议而不是类时,类适配器才能用Objective-C实现。

  对象的适配器模式:

  

  Target和Adapter之间的关系与类适配器相同,而Adapter和Adaptee之间的关系从“属于(继承)”变成了“包含(组合引用)”。这种情况下Adapter持有一个对Adaptee的引用,在request方法中,Adapter发送[adaptee specificRequest]消息,以间接方法Adaptee的specificRequest方法,然后实现客户端请求的其他部分。

三,使用场景:(也是copy的,多读读回忆下项目代码就能理解了)

  1:你想使用一个已经存在的类,而它的接口不符合你的需求,即已有类的接口与需求不匹配
  2:你想创建一个可以复用的类,该类可以与其他不相关的类或不可预见的类(即那些接口可能不一定兼容的类)协同工作
  3:(仅适用于对象Adapter)你想使用一些已存在的子类,但是不可能对每一个都进行子类化以匹配它们的接口,对象适配器可以适配它的父类接口

四,Demo演示:
  下面有个场景,已经存在一个UniversulAudioPlayer类,可以进行音频的播放。但是ViewController端需要的是一个遵守了AudioPlayerProtocol协议接口的类对象 AudioPlayerProtocol协议规范了ViewController调用接口,她和UniversulAudioPlayer已经实现的接口并不匹配。代码如下:

Target

protocol AudioPlayerProtocol {

    /// setting media source
///
/// - Parameter url: source url
/// - Returns: whether success
func prepareWithUrl(url: URL) -> Bool /// jump to playing time point
///
/// - Parameter time: the time point 0 - media.duration
/// - Returns: whether success
func jumpTo(time: TimeInterval) -> Bool /// increase valume,add 0.1 once, value0 - 1
///
/// - Returns: Voide
func increaseVolume() -> Void /// reduce Volume
///
/// - Returns: Void
func reduceVolume() -> Void /// start to play the audio
///
/// - Returns:
func play() -> Void /// pause the audio
///
/// - Returns:
func pause() -> Void /// resume to play
///
/// - Returns:
func resume() -> Void
}

Adapee

class UniversulAudioPlayer: NSObject {

    private var url: URL?
private var duration: TimeInterval = 0
private var volumeValue: Float = 0
private var timeValue: TimeInterval = 0 var volume: Float{
set{
if newValue <= 0 {
self.volumeValue = 0
}else if newValue >= 1{
self.volumeValue = 1.0
}else{
self.volumeValue = newValue
}
print("new volume: \(self.volumeValue)")
}
get{
return self.volumeValue
}
} var time: TimeInterval{
set{
if newValue >= 0.0 && newValue <= duration {
self.timeValue = newValue
print("new time: \(newValue)")
}
}
get{
return self.timeValue
}
} func audioWithUrl(url: URL) -> Bool {
print("setting url: \(url)")
self.duration = 300;
self.url = url;
return true
} func playAudio() {
if url != nil {
print("start\resume to play audio \(url!)")
}else{
print("play fail")
}
} func pauseAudio() {
if url != nil {
print("pause audio \(url!)")
}else{
print("play fail")
}
}
}

Client

class ViewController: UIViewController {

    var player: AudioPlayerProtocol!

    override func viewDidLoad() {
super.viewDidLoad()
// player = MyUniversulAudioPlayer()
player = TheUniversulAudioPlayer()
player.prepareWithUrl(url: URL(string: "http://www.y.com/music/a.mp3")!)
} //按钮事件 @IBAction func playBtnClicked(_ sender: Any) {
player.play()
} @IBAction func pauseBtnClicked(_ sender: Any) {
player.pause()
} @IBAction func resumeBtnClicked(_ sender: Any) {
player.resume()
} @IBAction func increaseVolumeBtnClicked(_ sender: Any) {
player.increaseVolume()
} @IBAction func reduceVolumeBtnClicked(_ sender: Any) {
player.reduceVolume()
} }

所以,就需要我们做一个适配器
1,类的适配器模式:

class MyUniversulAudioPlayer: UniversulAudioPlayer, AudioPlayerProtocol {
func prepareWithUrl(url: URL) -> Bool {
return super.audioWithUrl(url: url)
} func jumpTo(time: TimeInterval) -> Bool {
super.time = time
return true
} func increaseVolume() {
self.volume += 0.1
} func reduceVolume() {
self.volume -= 0.1
} func play() {
self.playAudio()
} func pause() {
self.pauseAudio()
} func resume() {
self.playAudio()
} }

2,对象的适配器模式

class TheUniversulAudioPlayer: AudioPlayerProtocol {

    private var universulPlayer = UniversulAudioPlayer()

    func prepareWithUrl(url: URL) -> Bool {
return universulPlayer.audioWithUrl(url: url)
} func jumpTo(time: TimeInterval) -> Bool {
universulPlayer.time = time
return true
} func increaseVolume() {
universulPlayer.volume += 0.1
} func reduceVolume() {
universulPlayer.volume -= 0.1
} func play() {
universulPlayer.playAudio()
} func pause() {
universulPlayer.pauseAudio()
} func resume() {
universulPlayer.playAudio()
} }

  

设计模式-(6)适配器 (swift版)的更多相关文章

  1. iOS可视化动态绘制八种排序过程(Swift版)

    前面几篇博客都是关于排序的,在之前陆陆续续发布的博客中,我们先后介绍了冒泡排序.选择排序.插入排序.希尔排序.堆排序.归并排序以及快速排序.俗话说的好,做事儿要善始善终,本篇博客就算是对之前那几篇博客 ...

  2. Swift版iOS游戏框架Sprite Kit基础教程下册

    Swift版iOS游戏框架Sprite Kit基础教程下册 试读下载地址:http://pan.baidu.com/s/1qWBdV0C 介绍:本教程是国内唯一的Swift版的Spritekit教程. ...

  3. Swift版音乐播放器(简化版),swift音乐播放器

    这几天闲着也是闲着,学习一下Swift的,于是到开源社区Download了个OC版的音乐播放器,练练手,在这里发扬开源精神, 希望对大家有帮助! 这个DEMO里,使用到了 AudioPlayer(对音 ...

  4. 快速排序OC、Swift版源码

    前言: 你要问我学学算法在工作当中有什么用,说实话,当达不到那个地步的时候,可能我们不能直接的感觉到它的用处!你就抱着这样一个心态,当一些APP中涉及到算法的时候我不想给其他人画界面!公司的项目也是暂 ...

  5. swift版的CircleView

    swift版的CircleView 效果图 源码 // // CircleView.swift // CircleView // // Created by YouXianMing on 15/10/ ...

  6. swift版的GCD封装

    swift版的GCD封装 说明 本人针对swift封装了GCD,包括GCDQueue,GCDGroup,GCDTimer以及GCDSemaphore,使用较为便利. 源码 https://github ...

  7. swift版的StringAttribute

    swift版的StringAttribute 效果 源码 https://github.com/YouXianMing/Swift-StringAttribute // // StringAttrib ...

  8. swift版的元组

    swift版的元组 说明 元组的内容并不多,使用的话跟普通变量类似,以下是测试源码: // // ViewController.swift // Tuples // // Created by You ...

  9. swift版的枚举变量

    swift版的枚举变量 swift的枚举类型跟普通的类是极为类似的,使用的时候,请不要以为他是一个常量,以下是测试用源码 // // ViewController.swift // SwiftEnum ...

  10. 关东升的iOS实战系列图书 《iOS实战:入门与提高卷(Swift版)》已经上市

             承蒙广大读者的厚爱我的 <iOS实战:入门与提高卷(Swift版)>京东上市了,欢迎广大读者提出宝贵意见.http://item.jd.com/11766718.html ...

随机推荐

  1. python基础——2(基本数据类型及运算符)

    目录 为何数据要区分类型? 一.数字类型 1.整型int 2.浮点型float 二.字符串str 三.列表类型list 四.字典类型 五.布尔类型 运算符的介绍 一.算术运算符 二.比较运算符 三.赋 ...

  2. Android开发——程序锁的实现(可用于开发钓鱼登录界面)

    1. 程序锁原理 1.1 实现效果: 在用户打开一个应用时,若此应用是我们业务内的逻辑拦截目标,那就在开启应用之后,弹出一个输入密码的界面,输入密码正确则进入目标应用.若不输入直接按返回键,则直接返回 ...

  3. 异常 Failed to bind NettyServer on /10.133.7.216:29105, cause: Failed to bind to: /0.0.0.0:29105

    "C:\Program Files\Java\jdk1.7.0_80\bin\java" -agentlib:jdwp=transport=dt_socket,address=12 ...

  4. iOS-runtime-根据类名推送到任意控制器,且实现属性传值

    // // WJRuntime.m // RuntimeSkip // // Created by tqh on 15/9/8. // Copyright (c) 2015年 tqh. All rig ...

  5. python基础知识--条件判断和循环

    一.输入输出 python怎么来接收用户输入呢,使用input函数,python2中使用raw_input,接收的是一个字符串,输出呢,第一个程序已经写的使用print,代码入下: 1 name=in ...

  6. 可拔插的 IOC 容器

    可拔插的 IOC 容器 于是我打算自己实现一个这样的 bean 容器. 但在实现之前又想到一个 feature: 不如把实现 bean 容器的方案交给使用者选择,可以选择使用 bean 容器,也可以就 ...

  7. [POJ2446] Chessboard(二分图最大匹配-匈牙利算法)

    传送门 把所有非障碍的相邻格子彼此连一条边,然后求二分图最大匹配,看 tot * 2 + k 是否等于 n * m 即可. 但是连边不能重复,比如 a 格子 和 b 格子 相邻,不能 a 连 b ,b ...

  8. POJ3041:Asteroids【二分图匹配】

    二分图的最大匹配=最小顶点覆盖(Konig定理)=最大独立集的补集最大匹配经典的三种模型  这题就是最小顶点覆盖,顺便这题留给我的经验就是调试的时候一定要细心细心再细心对模板的各个细节都要熟!! #i ...

  9. springboot整合mybatis,freemarker

    springboot 整合mybaits,,freemarker pom.xml文件 <?xml version="1.0" encoding="UTF-8&quo ...

  10. Spring Boot使用HandlerInterceptorAdapter和WebMvcConfigurerAdapter实现原始的登录验证

    HandlerInterceptorAdapter的介绍:http://www.cnblogs.com/EasonJim/p/7704740.html,相当于一个Filter拦截器,但是这个颗粒度更细 ...