随着即将对所有Apple Watch设备进行的watchOS 3更新,许多手表应用程序的性能将大大提高。 这主要归因于watchOS应用可以利用的新形式的后台执行,以便定期刷新其内容,并始终准备好供用户查看的最新信息。

在本教程中,我将向您展示如何在watchOS 3中利用新的WatchKit API使自己的应用程序使用这些新的后台任务。 重要的是要注意,即使您的应用程序设置正确以支持这些后台刷新,根据特定用户的配置,系统也可能不会允许您的应用程序在后台运行。

本教程要求您在OS X El Capitan或更高版本上运行Xcode 8。

1. 后台任务

watchOS 3中的第一个重要的新类是 WKRefreshBackgroundTask 类。 这是一个抽象类,您不应继承其子类或创建自己的实例。 它仅定义一个属性 userInfo ,它可以存储与后台任务有关的额外信息。 它还定义了一个方法 setTaskCompleted() ,您必须调用该方法以告诉系统任务已完成。 如果您不拨打电话,那么系统将在应用程序上花费最大的时间来尝试完成任务,从而浪费电池并损害其他应用程序的性能。

当您的应用程序应处理正在完成的后台任务时,系统将调用watchOS扩展委托的 handle(_:) 方法。 该方法的单个参数是一组 WKRefreshBackgroundTask 对象,您需要对其进行循环并进行相应的处理。

在WatchKit框架内,Apple提供了以下四个具体子类: watchOS将根据任务向您的扩展委托传递适当类的实例。

  • WKApplicationRefreshBackgroundTask 是一种常规的后台任务类型,不适用于特定的刷新类型。 这种类型最常见的用例是安排其他任务。 例如,您可以在每天的特定时间为应用程序安排后台刷新。 然后,将相应的 WKApplicationRefreshBackgroundTask 对象传递到扩展委托的 handle(_:) 方法中时,您可以创建一个后台网络任务来下载一些数据以刷新应用程序的内容。
  • WKSnapshotRefreshBackgroundTask 是一种任务类型,专门用于您的应用程序需要更新其快照时。 我们将在本教程的后面部分讨论快照。 您现在只需要知道一个应用程序的快照用作其启动映像,并且也可以在用户的​​扩展坞中显示(可通过按侧面按钮进行访问)。 请注意,这种类型的后台任务具有独特的任务完成方法,我们稍后还将讨论。
  • WKWatchConnectivityRefreshBackgroundTask 是一种任务类型,用于当您通过WatchConnectivity框架将数据从iPhone传输到手表时。 由于此框架提供了一种将复杂数据直接传输到手表的方法,因此,该相应的后台任务类型主要用于计划快照更新,以响应从iPhone接收到的数据。
  • WKURLSessionRefreshBackgroundTask 是后台联网任务完成后分配给扩展委托的任务类型。 在这种情况下,仍将调用为 URLSession 对象设置的委托方法,在这里需要适当地处理数据。 仅调用扩展委托的 handle(_:) 方法,以便您的应用程序可以响应网络操作完成。 WKURLSessionRefreshBackgroundTask 不允许您访问 URLSession 对象或已传输的数据。 它定义的唯一额外属性是一个称为 sessionIdentifier 的字符串,您可以使用它来确定已完成的网络过程。

您可以通过多种方式在watchOS 3中安排后台刷新。对于简单的应用程序或快照刷新,可以在任何 WKExtension 对象上使用以下两种方法。

scheduleBackgroundRefresh(withPreferredDate:userInfo:scheduledCompletion:)

完成后,此方法将 WKApplicationRefreshBackgroundTask 对象返回给您的扩展委托。 watchOS系统将尝试在 Date 参数指定的时间在后台唤醒您的扩展程序。

请注意,不能保证您在此准确时间在后台唤醒您的应用程序。 由于其他情况(例如,电池剩余电量,正在运行的其他应用程序以及正在运行的内存等),系统可能决定稍后再唤醒您的应用程序。

可通过传递给扩展委托的任何后台任务对象访问 userInfo 参数。 如果您只需要发送与当前刷新有关的信息,这将特别有用。 最后, scheduledCompletion 参数可以提供一个代码块,该代码块将在计划了刷新或由于某些错误而失败后运行。

scheduleSnapshotRefresh(withPreferredDate:userInfo:scheduledCompletion:)

此方法将 WKSnapshotRefreshBackgroundTask 对象返回到您的扩展委托。 此方法的参数与计划常规应用程序刷新时的参数相同。 此方法的唯一区别是您计划的刷新类型。

以下是一段示例代码,可在watchOS应用程序中的任何地方使用它来安排后台刷新:

WKExtension.shared().scheduleBackgroundRefresh(withPreferredDate: Date(timeIntervalSinceNow: 60 * 60), userInfo: nil) { (error: Error?) in
    if let error = error {
        print("Error occurred while scheduling background refresh: \(error.localizedDescription)")
 

创建其他后台任务

如前所述,将创建WKWatchConnectivityRefreshBackgroundTask来响应您通过WatchConnectivity API从配对的iPhone向手表发送数据的情况。

当您使用后台配置创建URLSession对象时,系统会自动调度最后一种后台任务WKURLSessionRefreshBackgroundTask 。 在您的监视应用程序中的任何位置执行此操作将使系统自动为您安排请求,并在完成后调用扩展代表的handle(_:)方法。 以下示例代码在示例类中包含两个函数。 第一个功能配置并启动后台下载任务。 第二个是在URLSessionDelegate协议中声明的委托方法。

class SomeClass: URLSessionDelegate {
    func beginDownloadTask() {
        let config = URLSessionConfiguration.background(withIdentifier: "exampleSessionIdentifier")
        let session = URLSession(configuration: config, delegate: self, delegateQueue: nil)
        let task = session.downloadTask(with: URL(string: "https://www.example.com/data")!)
        task.resume()
    func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
        print("Session did complete")
 

2.观看应用快照

在watchOS 3中,应用程序一览表已从Apple Watch中完全删除,取而代之的是坞站 。 按下手表上的侧面按钮即可访问该扩展坞,其中包含用户可以配置的任意数量的应用程序:

如果您的应用程序已放置在用户的扩展坞中,则您有责任使扩展坞中显示的可见“快照”保持最新状态。

当您需要通过将WKSnapshotBackgroundRefreshTask传递到扩展委托的handle(_:)方法中来更新快照时,会通知您的应用程序。 收到此通知时,除了在后台唤醒WatchKit扩展程序外,还在后台和屏幕外唤醒应用程序的根接口控制器。 在此过程中,将按此顺序调用以下方法:

  1. 根接口控制器的awake(withContext:)方法
  2. 根接口控制器的willActivate()方法
  3. 扩展委托的handle(_:)方法

在这三种方法中的任何时候,您都可以自定义应用程序界面的外观,以创建显示在扩展坞中的快照。

刷新快照的最后一步发生在扩展委托的handle(_:)方法中。 与其他后台任务类型一样,您必须告诉系统任务已完成。 但是,对于快照刷新,可以通过在WKSnapshotBackgroundRefreshTask对象上调用setTaskCompleted(restoredDefaultState:estimatedSnapshotExpiration:userInfo:)方法来WKSnapshotBackgroundRefreshTask 。 调用此方法后,watchOS会捕获应用程序当前界面的快照,并将其保留在内存中,直到下次刷新。 为了更好地理解此方法,让我们遍历其所有参数:

  • restoredDefaultState只是一个布尔值,指定您的快照是否属于应用程序的第一个界面控制器。 如果您已转换到另一个接口,则为该值传递false
  • estimatedSnapshotExpiration是一个Date对象,它指定系统何时下次应尝试刷新应用程序的快照。 如果不需要立即安排其他刷新,则可以传入新的Date.distantFuture类属性。
  • userInfo是自定义数据的集合,如果您立即计划另一个刷新,则将传递到下一个WKSnapshotBackgroundRefreshTask对象。

3.改进WatchConnectivity

watchOS 2和iOS 9中引入的WatchConnectivity框架也通过watchOS 3和iOS 10获得了一些新功能。

如果用户将应用程序的复杂功能放在表盘上,则一天之内最多可以将数据从手机传输到手表上50次。 就像在iOS 9和watchOS 2中一样,这是通过WCSession transferCurrentComplication(userInfo:)实例方法完成的。 新的watchOS 3和iOS 10是WCSession remainingComplicationUserInfoTransfers属性,它会告诉你有多少转账剩余的那一天。

在今年的平台更新中添加的另一个新的WCSession属性是hasContentPending属性,该属性只是一个布尔值,表示是否有数据需要在iPhone和Apple Watch之间传输。

总体而言,watchOS 3中的新API允许您在后台快速有效地刷新Apple Watch应用程序的内容。 随着越来越多的应用程序采用这些新的API,watchOS应用程序的性能通常将大大提高并创造更好的用户体验。

链接的GitHub存储库包含一个watchOS 3 Xcode项目,其中包含一些示例代码,这些示例代码展示了如何在自己的应用程序中计划和响应后台刷新。

与往常一样,请确保在下面的评论中留下您的评论和反馈。

翻译自: https://code.tutsplus.com/tutorials/whats-new-in-watchos-3-background-tasks--cms-27126

Swift Apple Watch后台任务执行机制 官方对 WatchKit 后台任务的说明: https://developer.apple.com/documentation/watchkit/running_watchos_apps_in_the_background 在执行后台任务的时候,系统并一定会在约定的时间执行,系统会根据系统情况统筹各个应用后台任务的执行时间。 用 sch... Background TasksBackground tasks are a way for you to keep your app’s interfaces up-to-date. Receiving a background task object from the system is your signal to perform specific types of o... 文章目录前言一、Wear OS是什么?二、准备阶段1. 你需要的2. Wear OSAndroid相比如何?3. 专用于 Wear OS 的软件包三、开发1. 在 Android Studio 上创建 Wear OS 项目2.1 创建用户界面布局2.2 设置清单3. 最常见的 Wear OS UI 组件4. 使用"BoxInsetLayout"5. 使用"SwipeDismissFrameLayout"6. 导航6.1 使用导航抽屉四、结束 Watch Connectivity是与iOS 9和watchOS 2一起发布的通信框架。它的主要目的是在Apple Watch应用程序及其父iOS应用程序之间轻松无缝地传输信息。 该框架提供了许多不同的功能。 几周前 , Jorge Costa撰写了有关在iOS和Apple Watch应用程序之间发送消息的功能。 在本教程中,我们将放大在后台传输数据。 发送消息的功能旨在用于... 硬件Watchdog基本上已经是嵌入式芯片的标配了,它可以在系统出现严重错误时(程序跑飞,死锁等)无法恢复系统时,重启系统.原理比较简单:用一个定时器作为看门狗硬件,开启后,要定时"喂狗"(计数重载),如果在规定时间内没有"喂狗",看门狗定时器就会进行硬件重启. 在Android系统中,除了硬件Watchdog,还实现了一个软件Watchdog,主要是为了弥补硬件Watchdog功能的单一性,它...