GCD延长后台执行

EX1

在没有使用GCD时,当app被按home键退出后,app仅有最多5秒钟的时候做一些保存或清理资源的工作。但是在使用GCD后,app最多有10分钟的时间在后台长久运行。这个时间可以用来做清理本地缓存,发送统计数据等工作。

让程序在后台长久运行的示例代码如下:

复制代码
// AppDelegate.h文件
@property (assign, nonatomic) UIBackgroundTaskIdentifier backgroundUpdateTask;

// AppDelegate.m文件
- (void)applicationDidEnterBackground:(UIApplication *)application
{
[self beingBackgroundUpdateTask];
// 在这里加上你需要长久运行的代码
[self endBackgroundUpdateTask];
}

  • (void)beingBackgroundUpdateTask
    {
    self.backgroundUpdateTask = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
    [self endBackgroundUpdateTask];
    }];
    }

  • (void)endBackgroundUpdateTask
    {
    [[UIApplication sharedApplication] endBackgroundTask: self.backgroundUpdateTask];
    self.backgroundUpdateTask = UIBackgroundTaskInvalid;
    }

GCD的用法:

复制代码
// 后台执行:
dispatch_async(dispatch_get_global_queue(0, 0), ^{
// something
});

// 主线程执行:
dispatch_async(dispatch_get_main_queue(), ^{
// something
});

// 一次性执行:
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// code to be executed once
});

// 延迟2秒执行:
double delayInSeconds = 2.0;
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, delayInSeconds * NSEC_PER_SEC);
dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
// code to be executed on the main queue after delay
});

// 自定义dispatch_queue_t
dispatch_queue_t urls_queue = dispatch_queue_create(“blog.devtang.com”, NULL);
dispatch_async(urls_queue, ^{
   // your code
});
dispatch_release(urls_queue);

// 合并汇总结果
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
// 并行执行的线程一
});
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
// 并行执行的线程二
});
dispatch_group_notify(group, dispatch_get_global_queue(0,0), ^{
// 汇总结果
});

EX2

对于ios7,苹果支持几种后台运行模式,backgroundTask,voip,后台播放音乐等,具体看官方文档就好。

我这边需要在后台跑一个长时间运行的计时器,所以就不能让app进入suspend状态。

很早以前听说可以通过后台播放音乐来实现,借鉴了一下,测试好几天,找出来了一个还比较靠谱的方案:

首先在

  • (void)applicationDidEnterBackground:(UIApplication *)application{

}

里面申请backgroundTask

[[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:nil]

实现一个可以运行几分钟的权限。

然后写一个计时器实时检测 backgroundTimeRemaining

  • (void)tik{

    if ([[UIApplication sharedApplication] backgroundTimeRemaining] < 61.0) {

    [[CKAudioTool sharedInstance] playSound];

    [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:nil];
    

    }

}

[[CKAudioTool sharedInstance] playSound];这段代码是去播放了一个无声的音乐,很关键的一点是

[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback withOptions:AVAudioSessionCategoryOptionMixWithOthers error:&error]

这样后台播放就不会影响到别的程序播放音乐了。

我这个计时器每分钟运行一次tik函数,如果发现后台运行时间小于一分钟了,就再去申请一个backgroundTask。

神奇的地方在于:backgroundTask不能在程序已经进入后台的时候申请,可以用一个播放音乐的假前台状态去申请,所以可以做到不断申请到权限,也就完成了长时间后台执行。

牛逼

IOS允许长时间在后台运行的情况有7种:

  audio

  VoIP

  GPS

  下载新闻

  和其它附属硬件进行通讯时

  使用蓝牙进行通讯时

  使用蓝牙共享数据时

  除以上情况,程序退出时可能设置短暂运行10分钟

  让程序退出后台时继续运行10分钟

  在XXAppDelegate中增加:UIBackgroundTaskIdentifier bgTask;

  - (void)applicationDidEnterBackground:(UIApplication *)application
  {
  bgTask = [application beginBackgroundTaskWithExpirationHandler:^{
  // 10分钟后执行这里,应该进行一些清理工作,如断开和服务器的连接等
  // …
  // stopped or ending the task outright.
  [application endBackgroundTask:bgTask];
  bgTask = UIBackgroundTaskInvalid;
  }];
  if (bgTask == UIBackgroundTaskInvalid) {
  NSLog(@”failed to start background task!”);
  }
  // Start the long-running task and return immediately.
  dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  // Do the work associated with the task, preferably in chunks.
  NSTimeInterval timeRemain = 0;
  do{
  [NSThread sleepForTimeInterval:5];
  if (bgTask!= UIBackgroundTaskInvalid) {
  timeRemain = [application backgroundTimeRemaining];
  NSLog(@”Time remaining: %f”,timeRemain);
  }
  }while(bgTask!= UIBackgroundTaskInvalid && timeRemain > 0); // 如果改为timeRemain > 5*60,表示后台运行5分钟
  // done!
  // 如果没到10分钟,也可以主动关闭后台任务,但这需要在主线程中执行,否则会出错
  dispatch_async(dispatch_get_main_queue(), ^{
  if (bgTask != UIBackgroundTaskInvalid)
  {
  // 和上面10分钟后执行的代码一样
  // …
  // if you don’t call endBackgroundTask, the OS will exit your app.
  [application endBackgroundTask:bgTask];
  bgTask = UIBackgroundTaskInvalid;
  }
  });
  });
  }
  - (void)applicationWillEnterForeground:(UIApplication *)application
  {
  // 如果没到10分钟又打开了app,结束后台任务
  if (bgTask!=UIBackgroundTaskInvalid) {
  [application endBackgroundTask:bgTask];
  bgTask = UIBackgroundTaskInvalid;
  }
  }
  
  
  后台时,如果某些代码你不希望执行,可以加以下条件:
   UIApplication *application = [UIApplication sharedApplication];
  if( application.applicationState == UIApplicationStateBackground) {
  return;
  }
  有的app虽然我们不允许通知,但还是会弹出消息,应该是设置了定时器,到某一时间就让程序后台运行一会,从服务器更新数据,然后显示出来。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
图像识别技术在病虫害检测中的应用是一个快速发展的领域,它结合了计算机视觉和机器学习算法来自动识别和分类植物上的病虫害。以下是这一技术的一些关键步骤和组成部分: 1. **数据收集**:首先需要收集大量的植物图像数据,这些数据包括健康植物的图像以及受不同病虫害影响的植物图像。 2. **图像预处理**:对收集到的图像进行处理,以提高后续分析的准确性。这可能包括调整亮度、对比度、去噪、裁剪、缩放等。 3. **特征提取**:从图像中提取有助于识别病虫害的特征。这些特征可能包括颜色、纹理、形状、边缘等。 4. **模型训练**:使用机器学习算法(如支持向量机、随机森林、卷积神经网络等)来训练模型。训练过程中,算法学习如何根据提取的特征来识别不同的病虫害。 5. **模型验证和测试**:在独立的测试集上验证模型的性能,以确保其准确性和泛化能力。 6. **部署和应用**:将训练好的模型部署到实际的病虫害检测系统中,可以是移动应用、网页服务或集成到智能农业设备中。 7. **实时监测**:在实际应用中,系统可以实时接收植物图像,并快速给出病虫害的检测结果。 8. **持续学习**:随着时间的推移,系统可以不断学习新的病虫害样本,以提高其识别能力。 9. **用户界面**:为了方便用户使用,通常有一个用户友好的界面,显示检测结果,并提供进一步的指导或建议。 这项技术的优势在于它可以快速、准确地识别出病虫害,甚至在早期阶段就能发现问题,从而及时采取措施。此外,它还可以减少对化学农药的依赖,支持可持续农业发展。随着技术的不断进步,图像识别在病虫害检测中的应用将越来越广泛。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值