1、首先我们会想到xcode的设置,如下图
image.png
2、然而这样设置了,APP进入后台后,如果蓝牙连接着就会一直通讯,如果没有通讯,系统也会将进程挂起,所以需要进行心跳包,保证数据通讯不断。
但假如进入后台后蓝牙断开了,APP就被挂起来了,所以需要申请一段时间让蓝牙有重新连接的机会。
我们可以申请短暂的时间(300秒左右),让定时器不停跑搜索蓝牙的方法。为了保证app不会被系统回收,超过时间就把定时器关闭,APP将进入悬挂状态,直至下次重新进入app,之前收集的数据则可以操作保存。
var backTask :UIBackgroundTaskIdentifier?
var backTimer: Timer?
var backCounter: Int = 0
var isenterBackground = false
//APP进入后台
func applicationDidEnterBackground(_ application: UIApplication) {
backTask = UIApplication.shared.beginBackgroundTask(expirationHandler: nil)
self.isenterBackground = true
if connectedPeripheral?.state == .disconnected && isOnForBLE{
self.scanTimerStart()
}
}
//普通申请后台时间
func scanTimerStart(){
if backTimer != nil {
backTimer?.invalidate()
}
backCounter = 0
backTimer = Timer(timeInterval: 2, target: self, selector: #selector(backgroundScan), userInfo: nil, repeats: true)
RunLoop.current.add(self.backTimer!, forMode: RunLoop.Mode.default)
}
//蓝牙扫描
@objc func backgroundScan(){
discoveredPeripherals.removeAll(keepingCapacity: false)
AppCache.shared.deviceArray.removeAll()
LWBluetoothManager.sharedInstance.startScanPeripherals()
if connectedPeripheral != nil && connectedPeripheral?.state != .connected && AppCache.shared.ishandleDisconnect == false{
LWBluetoothManager.sharedInstance.connectPeripheral(connectedPeripheral!, valuableUUID: [kServiceUUID: [kCCharacteristicUUID]])
}
backCounter += 2
if #available(iOS 10.0, *) {
if backCounter >= 298 {
backCounter = 0
backTimer?.invalidate()
}
}else{
if backCounter >= 180 {
backCounter = 0
backTimer?.invalidate()
}
if backTask != nil {
UIApplication.shared.endBackgroundTask(backTask!)
}
}
}
3、蓝牙断开的时候,开起定时器。
网友评论