一.如何开启后台播放功能且程序不会挂起
详情请查看Demo,如果您觉得对您有帮助的话,别忘了Star一下哟,如果有什么问题请指正。
当应用退到后台或者锁屏状态下,过段时间程序就会被挂起,这时我们就需要开启一个后台服务来防止后台程序挂起,我是通过一个定时器来开启服务的。直接上代码。
step1: 首先开启后台模式中的音频,如下图
step2: 在AppDelegate.m文件中导入头文件 #import <AVFoundation/AVFoundation.h>
#import <AVFoundation/AVFoundation.h>
@interface AppDelegate ()
@property (strong, nonatomic) NSTimer *backgroundTime;
@property (nonatomic, assign) UIBackgroundTaskIdentifier bgTask;
@end
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
return YES;
}
- (void)applicationWillResignActive:(UIApplication *)application {
//程序失去焦点的时候 开启后台服务
[self openServiceWhenBack];
}
- (void)applicationWillEnterForeground:(UIApplication *)application {
//当程序进入前台的时候结束后台任务
[self endBackgroundTask];
}
#pragma mark ---------------------------- 开启一个后台任务
- (void)openServiceWhenBack{
UIApplication* app = [UIApplication sharedApplication];
self.bgTask = [app beginBackgroundTaskWithExpirationHandler:^{
[app endBackgroundTask:self.bgTask];
self.bgTask = UIBackgroundTaskInvalid;
}];
//开启定时器 不断向系统请求后台任务执行的时间
self.backgroundTime = [NSTimer scheduledTimerWithTimeInterval:5 target:self selector:@selector(applyForMoreTime) userInfo:nil repeats:YES];
[self.backgroundTime fire];
}
//申请后台任务
-(void)applyForMoreTime {
//如果系统给的剩余时间小于60秒 就终止当前的后台任务,再重新初始化一个后台任务,重新让系统分配时间,这样一直循环下去,保持APP在后台一直处于active状态。
if ([UIApplication sharedApplication].backgroundTimeRemaining < 60) {
[[UIApplication sharedApplication] endBackgroundTask:self.bgTask];
self.bgTask = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
[[UIApplication sharedApplication] endBackgroundTask:self.bgTask];
self.bgTask = UIBackgroundTaskInvalid;
}];
}
}
#pragma mark ---------------------------- 结束一个后台任务
- (void)endBackgroundTask {
UIApplication *app = [UIApplication sharedApplication];
[app endBackgroundTask:self.bgTask];
self.bgTask = UIBackgroundTaskInvalid;
if (self.backgroundTime) {
// 结束计时
[self.backgroundTime invalidate];
self.backgroundTime = nil;
}
}
以上是开启后台服务的操作,这样就能避免程序被挂起
接下来说明一下如果有音乐播放器播放的时候如何播放自己的音频问题
step1: 初始化播放器(初始化的时候一定要把播放器设置为全局变量,否则会没有声音),这里我用的是懒加载初始化
///懒加载播放器
- (AVAudioPlayer *)voicePlayer{
if (!_voicePlayer) {
NSString *filePath = [[NSBundle mainBundle] pathForResource:@"defaultAudio" ofType:@"mp3"];
NSURL *fileUrl = [NSURL fileURLWithPath:filePath];
_voicePlayer = [[AVAudioPlayer alloc] initWithContentsOfURL:fileUrl error:nil];;
_voicePlayer.volume = 1;
//这里的代理如果要用到就可以设置,没有用到就不用设置
_voicePlayer.delegate = self;
}
return _voicePlayer;
}
播放音频
- (void)viewDidLoad {
[super viewDidLoad];
//播放音乐
[self.voicePlayer play];
}
停止播放音频
- (void)stopPlayVoiceAction{
//停止播放音频
[self.voicePlayer stop];
}
播放完成代理 (其他代理方法请自行处置,这里只举例播放完成事件)
#pragma mark ------------------------ AVAudioPlayer的代理事件
//播放完成
- (void)audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag{
if (flag) {
//播放完成后继续从头播放
[self.voicePlayer play];
}
}
step2: 后台播放音频及同时播放音乐播放器
这里就需要介绍一下AVAudioSession类,在这里请参考iOS 音频-AVAudioSession文章,感谢简书作者安东_Ace,这篇文章介绍的比较详细。
当程序进入后台的时候,我们需要设置AVAudioSession为后台模式并且为活跃状态,
[[AVAudioSession sharedInstance] setActive:YES error:nil];
[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback error:nil];
这个时候我们的音频文件就会一直在后台播放了。
但是这种情况下假如我们正在听音乐,当我们的音频播放的时候音乐就会被停止,如果想继续听音乐我们就的需要把音乐重新点击播放。那我们能不能像地图导航一样实现在能听音乐的时候还能收到语音播报呢,而且这是语音播报的声音被放大,音乐声音被调小,而且语音播报完成后就恢复正常音乐音量呢,答案是肯定的。
这个时候我们就需要设置AVAudioSession的options了。
//AVAudioSessionCategoryOptionMixWithOthers:允许其他音频文件同时播放
//AVAudioSessionCategoryOptionDuckOthers:当播放本音频时其他音频音量变小
[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback withOptions:AVAudioSessionCategoryOptionMixWithOthers | AVAudioSessionCategoryOptionDuckOthers error:nil];
如果音频播放完后,需要重新把音乐声音恢复,在音频播放完后或者自己想要恢复的时候
[[AVAudioSession sharedInstance] setActive:NO withOptions:AVAudioSessionSetActiveOptionNotifyOthersOnDeactivation error:nil];
这样就能把音乐的声音恢复了,类似地图导航。