【Swift开发专栏】Swift中的协议与委托模式

简介: 【4月更文挑战第30天】Swift编程语言强调协议与委托模式。协议定义了类型需实现的方法和属性,如`SomeProtocol`示例。遵循协议的类、结构体或枚举需实现协议要求。协议可继承,也可作为类型使用。委托模式让对象间通信更灵活,通过协议实现,如`DataSourceDelegate`示例。实战案例展示了在`UITableView`和自定义下载器中使用委托模式。

Swift 是一种强大而灵活的编程语言,它提供了一系列特性来支持面向协议编程。其中,协议(Protocol)和委托模式(Delegate Pattern)是两个核心概念,广泛应用于iOS和macOS应用程序开发中。本文将详细介绍Swift中的协议与委托模式,并通过三个部分来展开讨论。

第一部分:协议的基本概念和使用

1.1 协议的定义

协议是一种用于定义方法、属性和构造器的蓝图。它规定了遵循协议的类型必须实现的功能。在Swift中,协议使用protocol关键字定义。

protocol SomeProtocol {
   
    var mustBeSettable: Int {
    get set }
    var doesNotNeedToBeSettable: Int {
    get }
    func doSomething()
}

1.2 遵循协议

要让一个类、结构体或枚举遵循协议,需要在类型名称后加上协议名称,中间用冒号分隔。

class SomeClass: SomeProtocol {
   
    var mustBeSettable: Int = 0
    let doesNotNeedToBeSettable: Int = 0
    func doSomething() {
   
        print("Doing something!")
    }
}

1.3 协议的继承

协议可以继承一个或多个其他协议,从而组合多个协议的要求。

protocol InheritingProtocol: SomeProtocol, AnotherProtocol {
   
    // 协议定义
}

1.4 协议作为类型

在Swift中,协议本身也可以作为一种类型使用,这意味着可以将协议作为函数、方法或构造器的参数、返回值或属性类型。

func doSomethingWithProtocol(protocolInstance: SomeProtocol) {
   
    protocolInstance.doSomething()
}

第二部分:委托模式的概念和应用

2.1 委托模式简介

委托模式是一种设计模式,它允许一个对象将某些任务委托给另一个对象。在委托模式中,通常有一个委托者(Delegate)和一个委托对象(Delegating Object)。委托对象会持有一个委托者的引用,并在适当的时候调用委托者的方法。

2.2 委托模式的实现

在Swift中,委托模式通常通过协议来实现。委托者遵循协议,并提供必要的方法实现。委托对象持有一个符合协议的类型的引用,并在需要时调用相应的方法。

protocol DataSourceDelegate {
   
    func dataSourceUpdated()
}
class DataSource {
   
    var delegate: DataSourceDelegate?
    func updateData() {
   
        // 更新数据
        delegate?.dataSourceUpdated()
    }
}
class ViewController: DataSourceDelegate {
   
    let dataSource = DataSource()
    func viewDidLoad() {
   
        dataSource.delegate = self
    }
    func dataSourceUpdated() {
   
        print("Data source updated!")
    }
}

2.3 委托模式的优势

委托模式的优势在于它提供了一种解耦的方式,使得委托对象和委托者之间的交互更加灵活。它还允许一个对象将任务委托给多个不同的对象,从而实现更复杂的功能。

第三部分:实战案例

3.1 实战案例一:UITableView的委托和数据源

在iOS开发中,UITableView是一个常见的组件,它使用了委托和数据源模式。开发者需要创建一个遵循UITableViewDelegateUITableViewDataSource协议的类,并实现必要的方法。

class MyTableViewDelegate: NSObject, UITableViewDelegate {
   
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
   
        // 处理行点击事件
    }
}
class MyTableViewDataSource: NSObject, UITableViewDataSource {
   
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
   
        // 返回行数
        return 0
    }
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
   
        // 创建和配置单元格
        return UITableViewCell()
    }
}
class ViewController {
   
    let tableView = UITableView()
    let delegate = MyTableViewDelegate()
    let dataSource = MyTableViewDataSource()
    func viewDidLoad() {
   
        tableView.delegate = delegate
        tableView.dataSource = dataSource
    }
}

3.2 实战案例二:自定义委托模式

在某些情况下,开发者可能需要创建自己的委托模式。例如,一个下载器类可能需要一个委托来通知下载进度和完成状态。
```swift
protocol DownloaderDelegate {
func downloaderDidFinishDownloading(downloader: Downloader)
func downloader(downloader: Downloader, didUpdateProgress progress: Float)
}
class Downloader {
var delegate: DownloaderDelegate?
func startDownload() {
// 开始下载
// 更新进度
// 通知委托
}
}
class MyDownloaderDelegate: DownloaderDelegate {
func downloaderDidFinishDownloading(downloader: Downloader) {
print("Download finished!")
}
func downloader(downloader: Downloader, didUpdate

相关文章
|
18天前
|
数据可视化 数据处理 Swift
Swift开发——简单App设计
SwiftUI教程概述:简化App设计,通过代码展示了如何创建一个计算两个数之和的界面。工程`MyCh0902`包含`ContentView.swift`,其中定义了`ContentView`和`MyView`结构体。`MyView`负责界面布局,使用`VStack`和`HStack`组织元素,如`TextField`和`Button`。点击`Button`调用`calc`方法处理输入并更新结果。界面设计可在Xcode的Inspector窗口中可视化配置。推荐将界面逻辑移到单独的`MyView.swift`文件中以清晰分离视图设计。
186 1
Swift开发——简单App设计
|
26天前
|
存储 Swift 索引
Swift开发——索引器扩展
扩展用于向已存在的类型(例如,类、结构体、枚举和协议等)中添加新的功能,扩展甚至可以向系统类型(包括无法查阅代码的类型)中添加新的功能,但是扩展不能覆盖原类型中已有的方法,扩展也不能向类中添加新的存储属性。
37 6
Swift开发——索引器扩展
|
4天前
|
存储 安全 Swift
Swift高级特性:泛型与协议
【7月更文挑战第10天】Swift高级特性:泛型与协议增强代码复用与类型安全。泛型允许编写通用代码,如`swap`函数和泛型`Stack`结构体,支持类型约束如`Comparable`。协议定义行为蓝图,类型遵循协议需实现其要求。通过两者结合,构建高效灵活的代码结构。
|
26天前
|
存储 Swift
Swift开发——属性检查器
Swift中的属性检查器(willSet, didSet)允许在设置存储属性值前后执行代码。在类`Circle`中,属性`radius`使用属性观察器:willSet在赋值前检查值,若值为负则打印警告;didSet在赋值后比较新旧值,根据变化输出相应信息。在实例`c`中,`radius`从-5变为0时,输出“Input value is negative.”和“The circle gets smaller.”;从0变为10时,输出“Input value is normal.”和“The circle gets larger.”。
181 4
Swift开发——属性检查器
|
28天前
|
Swift C++ 索引
Swift开发——简单函数实例
函数是编程的基础,用于封装特定功能的代码。它们有关键词func、函数名、参数列表(可为空)和返回类型。多返回值可通过元组、数组、inout参数或可选类型实现。例如,返回元组 `(value1, value2)`,数组 `[value1, value2]` 或使用可选数组 `[[Double]]?`。函数可以作为其他函数的参数,类似闭包或Lambda表达式。在Swift中,示例展示了通过元组、带索引的元组、数组和可选类型返回多个值的函数。还演示了如何使用inout参数交换变量值。
28 5
Swift开发——简单函数实例
|
25天前
|
存储 Swift
Swift开发——弱占用
Swift的自动引用计数(ARC)管理类实例内存,通过强引用保持实例存活。当出现强引用循环时,可使用`weak`关键字创建弱引用,避免阻止实例释放。弱引用在不再被强引用时导致对象立即释放。示例中,添加`weak`至`author`和`book`变量防止引用循环,使得两者析构器均执行,释放内存。图2展示了弱引用结构,当解除所有强引用后,ARC自动释放实例,调用析构器。
184 1
Swift开发——弱占用
|
27天前
|
存储 程序员 Swift
Swift开发——存储属性与计算属性
Swift推荐使用结构体进行开发,结构体支持属性和方法,且作为值类型。结构体属性包括存储属性(如radius)和计算属性(如r),计算属性不存储值,类似方法。结构体用`struct`定义,命名遵循大驼峰规则。实例名遵循小驼峰规则。属性可在结构体中任意位置定义,静态属性用`static`。存储属性可为`lazy`实现懒加载。结构体实例通过`.`访问属性和方法,静态属性和方法用`结构体名.`访问。计算属性可读写,可通过`get`和`set`定义。程序段1展示了结构体Point和Circle的属性和方法,包括私有属性、只读计算属性、可读写计算属性及`mutating`方法。
13 0
Swift开发——存储属性与计算属性
|
29天前
|
存储 算法 Swift
Swift开发——循环执行方式
Swift语言中的循环主要包括`for-in`和`while`结构。`for-in`适用于遍历数字区间、字符串和字典,支持使用`stride`函数定制步进。字典遍历时,可以用二元元组`(k, v)`访问键值对。`while`循环有标准形式和`repeat-while`形式,确保至少执行一次循环体。程序示例展示了`for-in`和不同`while`结构的用法,包括计算阶乘、奇数和、加密字符串以及最大公约数和最小公倍数。
18 0
Swift开发——循环执行方式
|
1月前
|
Swift 索引
Swift开发——元组
Swift中的元组是一种数据结构,用于组合不同类型的值。它们不是独立的数据类型,而是以有序序列形式存在,用圆括号括起,元素间用逗号分隔。元组可以有任意数量和类型的元素,可变性取决于其定义。常用于函数返回多个值。示例代码展示了元组的创建、访问、解包及赋值。元组可以通过标签来标识元素,支持嵌套和比较。在函数返回值和并行赋值场景中,元组特别有用。
31 0
Swift开发——元组
|
1月前
|
安全 JavaScript Swift
Swift开发——输出格式化字符
这篇文章介绍了Swift语言的基本概念和格式化输出方法。Swift是苹果平台的官方编程语言,强调类型安全,使用`main.swift`作为程序入口。文章通过示例展示了如何使用`print`函数进行格式化字符串输出,包括控制整数和浮点数的宽度、对齐方式以及使用`String(format:)`函数。示例代码创建了一个名为Ch0001的工程,并展示了不同输出方式,如常规输出、格式化输出和使用`\(`常量名或变量名`)`的形式。最后,文章给出了程序的运行结果截图。
25 0
Swift开发——输出格式化字符