Swift中发布-订阅框架Combine的使用

swift,发布,订阅,框架,combine,使用 · 浏览次数 : 15

小编点评

**Combine 简介** Combine 是苹果新框架用于处理事件的响应式框架,支持 iOS 13 及以上版本。它提供了一种声明式、函数式的编程方式,让开发者更简单、高效地处理异步数据流。 **核心概念** * **Publisher**:发布者,可以将值或错误发送给订阅者。 * **Subscriber**:订阅者,接收发布者发送的值、错误和完成信息,并进行相应的处理。 * **Operator**:用于对数据流进行转换、过滤、合并等处理。 **主要功能** * 支持各种操作符,例如 `map`、`filter`、`merge`等。 * 提供 `cancel` 方法,允许取消订阅以释放资源。 * 支持多种订阅模式,例如 `sink`、`subscribe`、`on` 等。 **示例** ```swift let publisher = Just("hello, world!") let subscriber = publisher.map { $0.uppercased() } subscriber.sink { _ in print("received \(subscriber.value)") } ``` **优点** * 简化和简化代码。 * 提供声明式的编程方式。 * 支持多种订阅模式。 * 提供 `cancel` 方法,释放资源。 **用法** 1. 创建一个 `Publisher` 对象,用于发布数据。 2. 创建一个 `Subscriber` 对象,接收数据。 3. 使用 `map`、`filter`等操作符对数据流进行处理。 4. 使用 `cancel` 方法取消订阅。 5. 使用 `sink` 等方法监听数据流并处理数据。

正文

Combine简介
Combine是一个苹果用来处理事件的新的响应式框架,支持iOS 13及以上版本。
你可以使用Combine去统一和简化在处理类似于target-action,delegate,kvo等事情的代码。
iOS目前已经有第三方的响应式框架了,如:RxSwift、ReactiveCocoa,但是苹果现在发布了自己的新的框架。
它提供了一种声明式、函数式的编程方式,可以让开发者更加简单、高效地处理异步数据流。

Combine核心概念
Combine的核心概念是Publisher、Subscriber和Operator。发布者,订阅者和中间操作者。
发布者Publisher:可以将值或错误发送给订阅者,并且在完成或取消时,向订阅者发送相应的消息。
订阅者Subscriber:接收发布者发送的值、错误和完成信息,并进行相应的处理。
操作符Operator:用来对数据流进行转换、过滤、合并等处理,Combine框架中提供了很多操作符,例如map、filter、merge等。

当一个Publisher发布一个事件时,Subscriber会被通知这个事件,并执行相应的操作。
一个Subscriber可以订阅多个Publisher,并且每个Publisher都可以同时拥有多个Subscriber。当所有的Subscriber都取消订阅时,Publisher会停止发布事件。
Operator可以对事件流进行各种操作,包括数据转换、过滤、组合等。比如,map操作可以将一个事件流中的元素转换成另一个类型,filter操作可以过滤出符合条件的元素,merge操作可以将多个事件流合并成一个事件流等。
 
map操作符可以将数据流中的值进行转换,返回一个新的数据流。例如,将一个字符串转换成大写形式:
let stringPublisher = Just("hello, world!")
let uppercasedPublisher = stringPublisher.map { $0.uppercased() }
filter对数据流中的值进行过滤,返回一个新的数据流。例如,过滤掉数组中的偶数:
let arrayPublisher = [1, 2, 3, 4, 5].publisher
let oddPublisher = arrayPublisher.filter { $0 % 2 != 0 }
merge对数据流进行合并,生成一个数据流。例如,将两个发布者合并成一个:
let publisher1 = Just("hello")
let publisher2 = Just("world")
let mergedPublisher = Publishers.Merge(publisher1, publisher2)
下面是一个简单的例子,展示了如何使用Combine来监听一个UITextField的文本输入
import UIKit
import Combine

class ViewController: UIViewController {

    @IBOutlet weak var textField: UITextField!

    var cancellables = Set<AnyCancellable>()

    override func viewDidLoad() {
        super.viewDidLoad()

        // 创建一个订阅者,打印出文本输入的内容
        let subscriber = Subscribers.Assign(object: self.textField, keyPath: \.text)

        // 订阅文本输入框的文本变化事件
        self.textField.textPublisher(for: .editingChanged)
            .map { $0.text ?? "" }
            .subscribe(subscriber)
            .store(in: &cancellables)
    }
}

extension UITextField {
    func textPublisher() -> AnyPublisher<String, Never>
    {
        NotificationCenter.default
            .publisher(for: UITextField.textDidChangeNotification, object: self)
            .map {($0.object as? UITextField)?.text  ?? ""}
            .eraseToAnyPublisher()
    }
}

 

AnyCancellable
Combine 中还定义了一个 AnyCancellable 类,实现了 Cancellable 协议,特点是会在该类析构时自动执行 cancel 方法。

在实际开发中,当 Subscriber 在某个时候不想接收 Publisher 发布的数据时,可以取消订阅以释放资源。
Combine 中提供了 Cancellable 协议,该协议中定义了一个 cancel 方法,用于取消订阅流程。
import Combine
let subject = PassthroughSubject<String, Never>() // 创建PassthroughSubject
// 订阅
let subscription = subject.sink(receiveCompletion: { _ in
    print("receiveCompletion")
}, receiveValue: { value in
    print(value)
    
})
// 发送数据
subject.send("hello")
// 中途取消订阅
subscription.cancel()
// 后面发送的数据都会失败
subject.send("world")
subject.send(completion: .finished)
/* 输出:
 hello
 */
可以理解为 AnyCancellable是一种管理订阅状态的工具,能根据开发者需要在某个时段切断 Publisher和 Subscriber的联系。
AnyCancellable 的一个应用就是可以在某种情况下中断网络请求,实现如下:
import UIKit
import Combine
let dataPublisher = URLSession.shared.dataTaskPublisher(for: URL(string: "https://louyu.cc")!)
let cancellableSink = dataPublisher.sink { completion in
    switch completion {
    case .finished:
        break
    case .failure(let error):
        print("error: \(error)")
        break
    }
} receiveValue: { value in
    print("received \(value)")
}
cancellableSink.cancel() //取消网络请求

参考文章地址:
https://www.jianshu.com/p/1dc27229a533
https://louyu.cc/articles/ios-swift/2021/03/?p=2865/
 

与Swift中发布-订阅框架Combine的使用相似的内容:

Swift中发布-订阅框架Combine的使用

Combine简介 Combine是一个苹果用来处理事件的新的响应式框架,支持iOS 13及以上版本。 你可以使用Combine去统一和简化在处理类似于target-action,delegate,kvo等事情的代码。 iOS目前已经有第三方的响应式框架了,如:RxSwift、ReactiveCoc

Swift中UITableViewDiffableDataSource的使用

在 iOS 13 中 Apple 为 UITableView 和 UICollectionView 引入了 DiffableDataSource, 让开发者可以更简单高效的实现 UITableView、UICollectionView 的局部数据刷新。 新的刷新的方法为 apply 通过使用 app

Swift中常见的String用法,Array高阶使用,Set集合操作

String字符串常见用法 生成字符串 创建字符串 let greeting = "Hello, world!" let name = String("John") 连接字符串:使用加号(+)或者字符串插值(使用())来将多个字符串连接起来。 var firstName = "John" let l

Swift中指针UnsafePointer的常见用法

指针类型 //基本指针 UnsafePointer const T * UnsafeMutablePointer T * //集合指针 UnsafeBufferPointer const T * //指向一个连续已知类型区域,可以看成一个集合,并支持集合操作 UnsafeMutableBuff

Swift与OC混编

Swift调OC 在Swift项目中调用OC类中的方法需要有个{targetName}-Bridging-Header.h文件,在这个文件中导入OC要暴露给Swift的类。 {targetName}-Bridging-Header.h文件的创建有2种方式 1.自己手动创建,然后在配置文件的Objec

modulemap的使用方法

modulemap的作用 modulemap 文件是用来解决 C,Object-C,C++ 代码在 Swift 项目中集成的问题的。 在 Swift 项目中,如果需要使用 C,Object-C 或者 C++ 代码,需要将相应的头文件和源文件导入到项目中,并且需要手动管理它们之间的依赖关系。导致项目结

Swift之struct二进制大小分析

随着Swift的日渐成熟和给开发过程带来的便利性及安全性,京喜App中的原生业务模块和基础模块使用Swift开发占比逐渐增高。本次讨论的是struct对比Class的一些优劣势,重点分析对包体积带来的影响及规避措施。

Swift下Data处理全流程:从网络下载,数模转换,本地缓存到页面使用

Swift下将网络返回json数据转换成struct 假如网络请求返回的数据结构是一个深层嵌套的Json 首先要通过key-value取出这个json中的数据源 // 将返回的json字符串转Dictory let json = """ { "name": "jack", "age": 20, "d

Swift函数调用方式浅析

函数的调用机制 函数的调用机制是在函数调用时通过那种路径走到最终调用函数地址的机制。 在编程语言中,函数的调用机制有三种 1.静态调用:编译期就确定了函数内存地址,执行效率最高,还可以使用编译器优化如:inline函数内联提升执行效率。缺点:因为函数调用的内存地址在编译期已经确定,则无法支持继承等动

万字长文详解如何使用Swift提高代码质量

京喜APP最早在2019年引入了Swift,使用Swift完成了第一个订单模块的开发。之后一年多我们持续在团队/公司内部推广和普及Swift,目前Swift已经支撑了70%+以上的业务。通过使用Swift提高了团队内同学的开发效率,同时也带来了质量的提升,目前来自Swift的Crash的占比不到1%。在这过程中不断的学习/实践,团队内的Code Review,也对如何使用Swift来提高代码质量有更深的理解。