服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - IOS - iOS实现后台长时间运行

iOS实现后台长时间运行

2021-05-14 16:00macro小K IOS

这篇文章主要为大家详细介绍了iOS实现后台长时间运行,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

前言

一般app在按下home键被挂起后,这时app的 backgroundtimeremaining 也就是后台运行时间大约只有3分钟,如果在退出app后,过十几二十二分钟或者更长时间再回到app,app就会回到刚打开时的状态,也就是首页;有的项目在被挂起后需要在后台运行一段时间,使有足够的时间来完成与服务器对接的操作,或者需要一直运行的需求;如果需要,则在app被挂起后,申请后台,来延长后台运行时间。

app申请后台运行的方式有几种:

播放音乐

定位

newsstand downloads

fetch 等;

iOS实现后台长时间运行

这里主要说下后台播放无声音乐(其实是不播放),采用哪种方式,对应勾选上图;我的项目中有音频播放需求,如果没有,那就找一个播放音频的理由,或者用其他方式实现。

实现

这里采用了两个单例:电话监控(xktelmanager)、后台运行(xkbgrunmanager),电话监控可以忽略,视情况而用;采用单例是为了方便管理;

xktelmanager.h

?
1
2
3
4
5
6
7
8
9
10
11
#import <foundation/foundation.h>
 
@interface xktelmanager : nsobject
///是否在后台运行
@property (nonatomic,assign) bool inbackgroundrun;
+ (xktelmanager *)sharedmanager;
/**
 来电监听
 */
- (void)startmonitor;
@end

xktelmanager.m

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#import "xktelmanager.h"
#import "xkbgrunmanager.h"
#import <coretelephony/ctcallcenter.h>
#import <coretelephony/ctcall.h>
 
static xktelmanager *_sharedmanger;
@interface xktelmanager()
@property (nonatomic, strong) ctcallcenter *callcenter;
@end
@implementation xktelmanager
+ (xktelmanager *)sharedmanager{
  static dispatch_once_t oncetelsingle;
  dispatch_once(&oncetelsingle, ^{
    if (!_sharedmanger) {
      _sharedmanger = [[xktelmanager alloc]init];
    }
  });
  return _sharedmanger;
}
- (instancetype)init{
  self = [super init];
  if (self) {
    _inbackgroundrun = no;
  }
  return self;
}
#pragma mark -********* 监听电话相关
- (void)startmonitor {
  __weak typeof(self) weakself = self;
  _callcenter = [[ctcallcenter alloc] init];
  _callcenter.calleventhandler = ^(ctcall * call) {
    ///如果已经进入后台了,不做任何操作
    if (weakself.inbackgroundrun) {
      return;
    }
    ///app未进入后台
    if ([call.callstate isequaltostring:ctcallstatedisconnected]){
      nslog(@"电话 --- 断开连接");
      [[xkbgrunmanager sharedmanager] stopbgrun];
    }
    else if ([call.callstate isequaltostring:ctcallstateconnected]){
      nslog(@"电话 --- 接通");
    }
    else if ([call.callstate isequaltostring:ctcallstateincoming]){
      nslog(@"电话 --- 待接通");
      [[xkbgrunmanager sharedmanager] startbgrun];
    }
    else if ([call.callstate isequaltostring:ctcallstatedialing]){
      nslog(@"电话 --- 拨号中");
      [[xkbgrunmanager sharedmanager] startbgrun];
    }
    else {
      nslog(@"电话 --- 无操作");
    }
    
  };
}
@end

xkbgrunmanager.h

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#import <foundation/foundation.h>
 
@interface xkbgrunmanager : nsobject
+ (xkbgrunmanager *)sharedmanager;
 
/**
 开启后台运行
 */
- (void)startbgrun;
 
/**
 关闭后台运行
 */
- (void)stopbgrun;
@end

xkbgrunmanager.m

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
#import "xkbgrunmanager.h"
///循环时间
static nsinteger _circuladuration = 60;
static xkbgrunmanager *_sharedmanger;
@interface xkbgrunmanager()
@property (nonatomic,assign) uibackgroundtaskidentifier task;
///后台播放
@property (nonatomic,strong) avaudioplayer *playerback;
@property (nonatomic, strong) nstimer *timerad;
///用来打印测试
@property (nonatomic, strong) nstimer *timerlog;
@property (nonatomic,assign) nsinteger count;
@end
@implementation xkbgrunmanager{
  cfrunloopref _runloopref;
  dispatch_queue_t _queue;
}
+ (xkbgrunmanager *)sharedmanager{
  static dispatch_once_t oncerunsingle;
  dispatch_once(&oncerunsingle, ^{
    if (!_sharedmanger) {
      _sharedmanger = [[xkbgrunmanager alloc]init];
    }
  });
  return _sharedmanger;
}
/// 重写init方法,初始化音乐文件
- (instancetype)init {
  if (self = [super init]) {
    [self setupaudiosession];
    _queue = dispatch_queue_create("com.audio.inbackground", null);
    //静音文件
    nsstring *filepath = [[nsbundle mainbundle] pathforresource:@"****" oftype:@"mp3"];
    nsurl *fileurl = [[nsurl alloc] initfileurlwithpath:filepath];
    self.playerback = [[avaudioplayer alloc] initwithcontentsofurl:fileurl error:nil];
    [self.playerback preparetoplay];
    // 0.0~1.0,默认为1.0
    self.playerback.volume = 0.01;
    // 循环播放
    self.playerback.numberofloops = -1;
  }
  return self;
}
 
- (void)setupaudiosession {
  // 新建audiosession会话
  avaudiosession *audiosession = [avaudiosession sharedinstance];
  // 设置后台播放
  nserror *error = nil;
  [audiosession setcategory:avaudiosessioncategoryplayback withoptions:avaudiosessioncategoryoptionmixwithothers error:&error];
  if (error) {
    nslog(@"error setcategory avaudiosession: %@", error);
  }
  nslog(@"%d", audiosession.isotheraudioplaying);
  nserror *activeseterror = nil;
  // 启动audiosession,如果一个前台app正在播放音频则可能会启动失败
  [audiosession setactive:yes error:&activeseterror];
  if (activeseterror) {
    nslog(@"error activating avaudiosession: %@", activeseterror);
  }
}
 
/**
 启动后台运行
 */
- (void)startbgrun{
  [self.playerback play];
  [self applyforbackgroundtask];
  ///确保两个定时器同时进行
  dispatch_async(_queue, ^{
    self.timerlog = [[nstimer alloc] initwithfiredate:[nsdate date] interval:1 target:self selector:@selector(log) userinfo:nil repeats:yes];
    self.timerad = [[nstimer alloc] initwithfiredate:[nsdate date] interval:_circuladuration target:self selector:@selector(startaudioplay) userinfo:nil repeats:yes];
    _runloopref = cfrunloopgetcurrent();
    [[nsrunloop currentrunloop] addtimer:self.timerad formode:nsdefaultrunloopmode];
    [[nsrunloop currentrunloop] addtimer:self.timerlog formode:nsdefaultrunloopmode];
    cfrunlooprun();
  });
}
 
/**
 申请后台
 */
- (void)applyforbackgroundtask{
  _task =[[uiapplication sharedapplication] beginbackgroundtaskwithexpirationhandler:^{
    dispatch_async(dispatch_get_main_queue(), ^{
      [[uiapplication sharedapplication] endbackgroundtask:_task];
      _task = uibackgroundtaskinvalid;
    });
  }];
}
 
/**
 打印
 */
- (void)log{
  _count = _count + 1;
  nslog(@"_count = %ld",_count)
}
 
/**
 检测后台运行时间
 */
- (void)startaudioplay{
  _count = 0;
  dispatch_async(dispatch_get_main_queue(), ^{
    if ([[uiapplication sharedapplication] backgroundtimeremaining] < 61.0) {
      nslog(@"后台快被杀死了");
      [self.playerback play];
      [self applyforbackgroundtask];
    }
    else{
      nslog(@"后台继续活跃呢");
    }///再次执行播放器停止,后台一直不会播放音乐文件
    [self.playerback stop];
  });
}
 
/**
 停止后台运行
 */
- (void)stopbgrun{
  if (self.timerad) {
    cfrunloopstop(_runloopref);
    [self.timerlog invalidate];
    self.timerlog = nil;
    // 关闭定时器即可
    [self.timerad invalidate];
    self.timerad = nil;
    [self.playerback stop];
  }
  if (_task) {
    [[uiapplication sharedapplication] endbackgroundtask:_task];
    _task = uibackgroundtaskinvalid;
  }
}
 
@end

最后在 appdelegate.m 对应的方法中,实现开启和停止后台运行即可!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://www.cnblogs.com/wangkejia/archive/2018/10/11/9773285.html

延伸 · 阅读

精彩推荐
  • IOS解析iOS开发中的FirstResponder第一响应对象

    解析iOS开发中的FirstResponder第一响应对象

    这篇文章主要介绍了解析iOS开发中的FirstResponder第一响应对象,包括View的FirstResponder的释放问题,需要的朋友可以参考下...

    一片枫叶4662020-12-25
  • IOSiOS通过逆向理解Block的内存模型

    iOS通过逆向理解Block的内存模型

    自从对 iOS 的逆向初窥门径后,我也经常通过它来分析一些比较大的应用,参考一下这些应用中某些功能的实现。这个探索的过程乐趣多多,不仅能满足自...

    Swiftyper12832021-03-03
  • IOS关于iOS自适应cell行高的那些事儿

    关于iOS自适应cell行高的那些事儿

    这篇文章主要给大家介绍了关于iOS自适应cell行高的那些事儿,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的...

    daisy6092021-05-17
  • IOSiOS中tableview 两级cell的展开与收回的示例代码

    iOS中tableview 两级cell的展开与收回的示例代码

    本篇文章主要介绍了iOS中tableview 两级cell的展开与收回的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    J_Kang3862021-04-22
  • IOSIOS开发之字典转字符串的实例详解

    IOS开发之字典转字符串的实例详解

    这篇文章主要介绍了IOS开发之字典转字符串的实例详解的相关资料,希望通过本文能帮助到大家,让大家掌握这样的方法,需要的朋友可以参考下...

    苦练内功5832021-04-01
  • IOSIOS 屏幕适配方案实现缩放window的示例代码

    IOS 屏幕适配方案实现缩放window的示例代码

    这篇文章主要介绍了IOS 屏幕适配方案实现缩放window的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要...

    xiari5772021-06-01
  • IOSiOS布局渲染之UIView方法的调用时机详解

    iOS布局渲染之UIView方法的调用时机详解

    在你刚开始开发 iOS 应用时,最难避免或者是调试的就是和布局相关的问题,下面这篇文章主要给大家介绍了关于iOS布局渲染之UIView方法调用时机的相关资料...

    windtersharp7642021-05-04
  • IOSiOS 雷达效果实例详解

    iOS 雷达效果实例详解

    这篇文章主要介绍了iOS 雷达效果实例详解的相关资料,需要的朋友可以参考下...

    SimpleWorld11022021-01-28