本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 —— 4.在 Combine 中使用计时器

内容概览

  • 前言
  • 使用计时器执行周期性的工作
  • 将计时器转换为计时器发布者(Timer.TimerPublisher)
  • 总结

前言

计时器是苹果开发者常用的功能。如果你的应用使用 Foundation 框架中的计时器 Timer 来周期性地执行某些操作,你可以用 Combine 简化这些代码。

现在,让我们来学习如何使用 Combine 处理计时器,并将已有的计时器处理代码迁移到 Combine

使用计时器执行周期性的工作

对于 iOS 开发人员而言,以下代码一定非常眼熟:

var timer: Timer?

override func viewDidLoad() {
super.viewDidLoad() timer = Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { _ in
self.myDispatchQueue.async() {
self.myDataModel.lastUpdated = Date()
}
}
}

以上代码使用 scheduledTimer(withTimeInterval:repeats:block:) 来实现每秒钟在 myDispatchQueue 中更新 self.myDataModel.lastUpdated 的功能。

将计时器转换为计时器发布者(Timer.TimerPublisher)

要将以上代码迁移到 Combine,只需将 TimerscheduledTimer(withTimeInterval:repeats:block:)的返回值) 替换为 Timer.TimerPublisher。调用 Timer. publish(every:tolerance:on:in:options:) 方法即可创建一个发布者。

每次底层的计时器(Timer)触发时,发布者都会发出一个新的日期(Date)实例,该日期代表计时器触发的瞬间。然后,你可以将 Combine 操作符应用到这个日期实例上,最终将这个发布者和一个订阅者(如:sink(receiveValue:)assign(to:on:))连接。

由于 Timer.TimerPublisher 遵从 ConnectablePublisher 协议,因此在您显式地连接之前,它不会产生任何元素。为此,可以通过手动调用 connect() 或使用 autoconnect() 运算符在订阅者连接时自动连接来实现。关于 ConnectablePublisher 的用法,可以参考 这篇文章

下一个示例将展示如何使用 Timer.TimerPublisher 替换上一个示例。它使用 Combine 的操作符来完成上一个示例中的闭包中的操作:

var cancellable: Cancellable?

override func viewDidLoad() {
super.viewDidLoad() cancellable = Timer.publish(every: 1, on: .main, in: .default)
.autoconnect()
.receive(on: myDispatchQueue)
.assign(to: \.lastUpdated, on: myDataModel)
}

在这个例子中,Combine 操作符替换了上一个示例的闭包中的所有行为:

  • receive(on:options:) 操作符确保了后续操作在指定的调度队列中执行,它替代了前面用到的 async() 调用;
  • assign(to:on:) 操作符通过键路径来更新数据模型的 lastUpdated 属性;

使用 Combine 来简化你的代码时,你会发现 Timer.TimerPublisher 会产生新的 Date 实例作为其输出类型。而第一个示例的闭包是将 Timer 本身作为其参数,因此它必须手动创建新的 Date 实例。

总结

使用 Combine 来简化你的计时器代码时,你会发现:

  • 代码易读性明显提升;
  • 线程切换变得更简单;
  • 数据模型的更新可以通过键路径(key path)来简化;

朋友,行动起来吧!把现有项目中的旧代码重构成使用 Combine 的代码~

本文内容来源:

Replacing Foundation Timers with Timer Publishers

Combine 框架,从0到1 —— 4.在 Combine 中使用计时器的更多相关文章

  1. Combine 框架,从0到1 —— 1.核心概念

      本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 1.核心概念.     内容概览 前言 核心概念 RxSwift Combine 总结 参考内容 ...

  2. Combine 框架,从0到1 —— 2.通过 ConnectablePublisher 控制何时发布

      本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 2.通过 ConnectablePublisher 控制何时发布.   内容概览 前言 使用 ma ...

  3. Combine 框架,从0到1 —— 3.使用 Subscriber 控制发布速度

      本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 3.使用 Subscriber 控制发布速度.   内容概览 前言 在发布者生产元素时消耗它们 使 ...

  4. Combine 框架,从0到1 —— 4.在 Combine 中使用通知

      本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 4.在 Combine 中使用通知.   内容概览 前言 让通知处理代码使用 Combine 总结 ...

  5. Combine 框架,从0到1 —— 4.在 Combine 中使用 KVO

      本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 4.在 Combine 中使用 KVO.   内容概览 前言 用 KVO 监控改动 将 KVO 代 ...

  6. Combine 框架,从0到1 —— 4.在 Combine 中执行异步代码

    本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 4.在 Combine 中执行异步代码. 内容概览 前言 用 Future 取代回调闭包 用输出类型( ...

  7. Combine 框架,从0到1 —— 5.Combine 提供的发布者(Publishers)

    本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 5.Combine 提供的发布者(Publishers). 内容概览 前言 Just Future D ...

  8. Combine 框架,从0到1 —— 5.Combine 中的 Subjects

    本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 5.Combine 中的 Subjects. 内容概览 前言 PassthroughSubject C ...

  9. Combine 框架,从0到1 —— 5.Combine 常用操作符

    本文首发于 Ficow Shen's Blog,原文地址: Combine 框架,从0到1 -- 5.Combine 常用操作符. 内容概览 前言 print breakpoint handleEve ...

随机推荐

  1. 2020-08-02:输入ping IP 后敲回车,发包前会发生什么?

    福哥答案2020-08-02: 首先根据目的IP和路由表决定走哪个网卡,再根据网卡的子网掩码地址判断目的IP是否在子网内.如果不在则会通过arp缓存查询IP的网卡地址,不存在的话会通过广播询问目的IP ...

  2. Linux学习日志——基本指令②

    文章目录 Linux学习日志--基本指令② 前言 touch cp (copy) mv (move) rm vim 输出重定向(> 或 >>) cat df(disk free) f ...

  3. 基于Linux系统geth的安装

    转载地址 https://blog.csdn.net/qq_36124194/article/details/83658580 基于Linux系统geth的安装 安装ethereum sudo apt ...

  4. Ansible常用模块-yum模块

    yum模块 name 必选 指定安装包名 state 执行命令  present  installed removed latest absent 其中installed and present等效 ...

  5. CPF 入门教程 - 样式和动画(三)

    CPF NetCore跨平台UI框架 系列教程 CPF 入门教程(一) CPF 入门教程 - 数据绑定和命令绑定(二) CPF 入门教程 - 样式和动画(三) 用样式可以对内部元素进行批量设置属性. ...

  6. vue cli3.0使用svg全过程

    VUE-cli3使用 svg-sprite-loader svg-sprite-loader 的插件,用来根据导入的 svg 文件自动生成 symbol 标签并插入 html 1.安装依赖 npm i ...

  7. xshell乱码解决办法

    https://jingyan.baidu.com/article/7908e85c758a58af481ad2ae.html

  8. Typescript node starter 1.Express Typescript

    启动项目 Express 是一个nodejs框架,用于构建Web后端应用程序.它非常的灵活,你可以用你喜欢的方式去使用他.在这个系列文章里,记录了我使用typescript express去构建一个w ...

  9. 跟我一起学.NetCore之配置变更监听

    前言 通常程序中配置少不了,配置的修改也避免不了,配置的热更新为此给应用程序带来很大的便捷,不用重启,提高用户体验:但往往有时候需要对修改进行审计,也就是需要记录,有时候也会针对配置修改的时候触发相关 ...

  10. 10.oracle分页

    oracle的分页一共有三种方式 方法一 根据rowid来分 SELECT * FROM EMP WHERE ROWID IN (SELECT RID FROM (SELECT ROWNUM RN, ...