美文网首页
Swift保持后台运行的几种机制

Swift保持后台运行的几种机制

作者: 怀心逝水 | 来源:发表于2019-08-03 17:12 被阅读0次

    什么是后台

    当我们在手机上按下Home或是锁屏键后,APP就会进入后台工作;或是切换到其他的APP界面的时候,之前的APP也会进入到后台的运行模式下。

    为什么需要后台运行

    APP正常进入后台后,在没有任何后台申请权限的状态下,只能存活大约200s的运行时间,之后程序将自动被系统挂起,这时候APP将不会处理程序的任何数据请求和逻辑处理;但有时候我们需要程序进入后台后一直保持运行状态,这时候就需要做什么呢?

    流程

    当离开APP触动的方法:func applicationDidEnterBackground(_ application: UIApplication);
    当快要进入APP时的触动方法:func applicationWillEnterForeground(_ application: UIApplication)
    当程序将要进入后台(墓碑状态):func applicationWillResignActive(_ application: UIApplication)

    场景

    1.需要APP退出后进入到后台模式下,并一直处于运行的状态中。
    2.通过消息推送的方式触点式的激活程序运行。

    我这边遇到的开发需求是:当用户切换或锁屏的状态下,当APP中的个人健康的数据达到报警的情况下,需要APP播放警报语音并且在APP中刷新用户的当前数据。

    那么我这边采用的是两种方式:
    一是APP进入后台以后,程序会一直申请资源运行,并且每十分钟进行一次数据请求,采集用户的个人数据,并作出相应的操作;
    二是通过推送消息(服务器那边会做报警的推送),APP端通过fetch消息,然后激活程序。

    权限申请

    image.png

    说明:App plays audio or streams audio/video using AirPlay申请APP后台的音频流的运行权限;
    App downloads content from the network申请APP后台的推送唤起的运行权限;

    那么主要看看代码吧:

        var count = 0
        var timer: Timer?
        var bgTask: UIBackgroundTaskIdentifier?
        
        func applicationDidEnterBackground(_ application: UIApplication) {
                    
            timer = Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(timerAction), userInfo: nil, repeats: true)
            RunLoop.current.add(timer!, forMode: .commonModes)
            bgTask = application.beginBackgroundTask(expirationHandler: nil)
        }
    

    一个定时器,主要是在程序快要挂起的前几十秒的时候,重新向系统请求,并且手动模拟的方式进入后台模式。

        @objc func timerAction() {
            
            count = count + 1
            print(count)
            if UIApplication.shared.backgroundTimeRemaining < 60.0 {
                
                let application = UIApplication.shared
                bgTask = application.beginBackgroundTask(expirationHandler: nil)
            }
            if count % (10*60) == 0 {
                //需要做的东西
            }
        }
    

    这样就循环重复的运行下去,当然这样只能程序一直运行,但是如果需要播放语音,这个时候需要怎么办呢?

        func applicationWillResignActive(_ application: UIApplication) {
            
            application.beginReceivingRemoteControlEvents()
            let session = AVAudioSession.sharedInstance()
            do {
                try session.setActive(true)
                try session.setCategory(AVAudioSessionCategoryPlayback)
            }catch {
                print(error)
            }
        }
    

    这样在进入后台模式下依然可以申请到语音播放的权限。

    这样就完成了第一种方式下的方法了。
    那么看看第二种方式:
    权限已经申请了,主要是在该APP下的推送时怎么激活程序。

    服务器推送消息需要做的工作

    在推送消息体里面必须包含"content-available"项并且设置值为"1"。

        func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable : Any], fetchCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
            
            RSHelper.refresh()  //需要做的操作
            completionHandler(UIBackgroundFetchResult.newData);
        }
    

    当然当APP激活进入前台的时候,后台的一切工作就需要禁止的,这个时候需要

        func applicationWillEnterForeground(_ application: UIApplication) {
            
            endBackTask()
        }
        
        func endBackTask() {
            
            timer?.invalidate()
            timer = nil
            count = 0
            UIApplication.shared.endBackgroundTask(bgTask!)
            bgTask = UIBackgroundTaskInvalid
        }
    

    最后两种方式下的方法介绍完毕

    相关文章

      网友评论

          本文标题:Swift保持后台运行的几种机制

          本文链接:https://www.haomeiwen.com/subject/vadiqctx.html