iOS AvPlayer AvAudioPlayer音頻的后台播放問題
首先,播放音頻之前先要設置AVAudioSession模式,通常只用來播放的App可以設為AVAudioSessionCategoryPlayback即可。模式意義及其他模式請參考文檔。
- AVAudioSession *session = [AVAudioSession sharedInstance];
- [session setCategory:AVAudioSessionCategoryPlayback error:nil];
- [session setActive:YES error:nil];
1。通知OS該app支持background audio。缺省情況下,當按下home鍵時,當前正在運行的程序被suspend,狀態從active變成in-active,也就是說如果正在播放音頻,按下HOME后就會停止。這里需要讓app在按在HOME后,轉到后台運行而非被suspend,解決辦法是在程序的-info.plist中增加required background modes這個key項,並選擇App plays audio這個value項。
2。現在按下HOME鍵后,程序退到后台,但是聲音仍在播放。但是如果要實現播放列表的依次播放、循環播放,即放完一首后自動切換到下一首,問題來了,當App在后台放完一首后,就會停下來。原因是在后台運行時,一旦聲音停下來,程序也隨之suspend,因此在切換文件加載的間隙,程序就會被suspend。曾經有山寨的解決辦法是專門起一個player的實例連續不停的放同一無聲音片斷,阻止程序被suspend。這里提供的方法是通過申請后台taskID達到后台切換播放文件的功能。
即聲明后台task id,並通過beginBackgroundTaskWithExpirationHandler將App設為后台Task,達到持續后台運行的目的。我們知道一般情況下,按HOME將程序送到后台,可以有5或10秒時間可以進行一些收尾工作,具體時間[[UIApplication sharedApplication] backgroundTimeRemaining]返回值。超時后app會被suspend,現在要做的就是用[[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:NULL]開始后台任務,可以將后台運行超時時間長時間的延長,具體延長多少時間還是見返回值,總之對於放段時間音樂應該夠了。另一個問題是每個開始的后台任務,都必須用endBackgroundTask來結束。 因此,在每次開始播放后啟動新的后台任務,同時結束上一個后台任務:
首先,要在viewdidload中
- [UIApplication sharedApplication] beginReceivingRemoteControlEvents];
- UIBackgroundTaskIdentifier newTaskId = UIBackgroundTaskInvalid;
- [avPlayer play];
- newTaskId = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:NULL];
- if (newTaskId != UIBackgroundTaskInvalid && oldTaskId != UIBackgroundTaskInvalid) {
- [[UIApplication sharedApplication] endBackgroundTask: oldTaskId];}
- oldTaskId = newTaskId;
當然,還有更方便的辦法就是在resignActive時beginBackgroundTaskWithExpirationHandler:並在BecomeActive中endBackgroundTask:
3。我們知道,ipod播放程序在后台時,雙擊HOME鍵,會有個控制界面,可以對它進行播放控制。
如果您想讓您的app可以像ipod一樣在后台也可以方便的通過雙擊HOME鍵來控制,就要用到遠程控制事件了。
首先在viewdidload等初始化的地方聲明App接收遠程控制事件,並在相應地方結束聲明
- - (void) viewWillAppear:(BOOL)animated
- {
- [super viewWillAppear:animated];
- [UIApplication sharedApplication] beginReceivingRemoteControlEvents];
- [self becomeFirstResponder];
- }
- - (void) viewWillDisappear:(BOOL)animated
- {
- [super viewWillDisappear:animated];
- [UIApplication sharedApplication] endReceivingRemoteControlEvents];
- [self resignFirstResponder];
- }
定義
- - (BOOL)canBecomeFirstResponder
- {
- return YES;
- }
最后定義 remoteControlReceivedWithEvent,處理具體的播放、暫停、前進、后退等具體事件
- - (void) remoteControlReceivedWithEvent: (UIEvent *) receivedEvent {
- if (receivedEvent.type == UIEventTypeRemoteControl) {
- switch (receivedEvent.subtype) {
- case UIEventSubtypeRemoteControlTogglePlayPause:
- [self playButtonPressed:playButton];
- [self testing];
- break;
- case UIEventSubtypeRemoteControlPreviousTrack:
- [self rewButtonReleased:(UIButton *)rewButton];
- break;
- case UIEventSubtypeRemoteControlNextTrack:
- [self ffwButtonReleased:(UIButton *)ffwButton];
- break;
- default:
- break;
- }
- }
- }
4. 至此,您有播放App已經相當完美了,還有最后一個問題,那就是當用戶使用耳機時,問題又來了。系統默認當插入耳機時,正在播放的聲音不中斷,直接切換到耳機播放,而當拔出耳機時,播放停止。如果這種行為滿足您的要求,那OK,否則您就需要進一步研究耳機檢測和聲音路由切換的問題。