這次主要是總結(jié)和記錄下視頻播放遇到的坑扇雕,視頻播放采用的是AVPlayer這個(gè)控件,語(yǔ)法大致如下:
NSURL * url = [NSURL fileURLWithPath:@"視頻地址"];
AVPlayerItem *playerItem = [AVPlayerItem playerItemWithURL:url];
self.player = [AVPlayer playerWithPlayerItem:playerItem];
[playerItem addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];
self.player.actionAtItemEnd = AVPlayerActionAtItemEndNone;
self.playerLayer = [AVPlayerLayer playerLayerWithPlayer:self.player];
self.playerLayer.videoGravity = AVLayerVideoGravityResizeAspect;
self.playerLayer.frame = self.view.bounds;
[self.view.layer addSublayer:self.playerLayer];
一般說(shuō)來(lái)优烧,這里要監(jiān)聽(tīng)AVPlayerItem的status屬性:
*
*AVPlayerItem的三種狀態(tài)
*AVPlayerItemStatusUnknown,
*AVPlayerItemStatusReadyToPlay,
*AVPlayerItemStatusFailed
*/
如果是AVPlayerStatusFailed說(shuō)明視頻加載失敗,這時(shí)可以通過(guò)self.player.error.description屬性來(lái)找出具體的原因趋箩。 問(wèn)題一:當(dāng)status變?yōu)锳VPlayerStatusReadyToPlay后赃额,我們調(diào)用play方法真的就能保證視頻正常播放嗎?
眾所周知叫确,AVPlayer支持的視頻跳芳、音頻格式非常廣泛,拋開(kāi)那些無(wú)法正常編解碼的情況竹勉,在某些情況下其可能就是無(wú)法正常播放飞盆。AVPlayer在進(jìn)行播放時(shí),會(huì)預(yù)先解碼一些內(nèi)容次乓,而此時(shí)如果我們的App使用CPU過(guò)多吓歇,I/O讀寫(xiě)過(guò)多時(shí),有可能導(dǎo)致視頻播放聲/畫(huà)不同步票腰,這點(diǎn)尤其在iPhone4上面表現(xiàn)更為明顯,用戶(hù)反饋iOS9.3.2的系統(tǒng)上也很明顯城看。而如果是發(fā)生在AVPlayer初始化解碼視頻的時(shí)候,有可能導(dǎo)致視頻直接無(wú)法播放杏慰,這時(shí)测柠,我們?cè)僬{(diào)用play或者seekToTime:方法都無(wú)法正常播放。建議不要在CPU或者I/O很頻繁的情況下使用AVPlayer,例如剛登錄App加載各種數(shù)據(jù)的情況下缘滥,可以等App預(yù)熱以后再使用鹃愤。
問(wèn)題二:當(dāng)rate屬性的值大于0后,真的就在播放視頻了嗎完域?
當(dāng)然不是。當(dāng)發(fā)生上面所講的情況時(shí)瘩将,我打印了當(dāng)前的rate情況吟税,是大于0的,但是頁(yè)面上顯示的情況卻還是什么也沒(méi)有姿现。有時(shí)候我們?nèi)绻胍谝曨l一播放的時(shí)候去做一些事情肠仪,例如設(shè)置一下播放器的背景色,如果我們僅僅是監(jiān)聽(tīng)這個(gè)rate可能無(wú)法100%保證有效备典,而如果我們真的要監(jiān)聽(tīng)這種情況的話(huà)异旧,有一個(gè)取巧的方法:
id _timerObserver = [self.player addBoundaryTimeObserverForTimes:@[[NSValue valueWithCMTime:CMTimeMake(1, 30)]] queue:dispatch_get_main_queue()
usingBlock:^{
//do something
}];
另外如果不需要監(jiān)聽(tīng)播放進(jìn)度的時(shí)候可以調(diào)下面的方法:
[self.player removeTimeObserver:_timerObserver];
問(wèn)題三:AVPlayer前后臺(tái)播放
當(dāng)我們切換到后臺(tái)后,這時(shí)AVPlayer通常會(huì)自動(dòng)暫停提佣,當(dāng)然如果設(shè)置了后臺(tái)播放音頻的話(huà)吮蛹,是可以在后臺(tái)繼續(xù)播放聲音的,正如蘋(píng)果自己的WWDC這個(gè)App一樣拌屏。這個(gè)功能在我的另一篇文章iOS AVPlayer之后臺(tái)連續(xù)播放視頻中解決了這個(gè)問(wèn)題潮针。
問(wèn)題四:音頻通道的搶占引起的無(wú)法播放視頻問(wèn)題
這個(gè)問(wèn)題下周我會(huì)另開(kāi)一篇博客專(zhuān)門(mén)講述,今兒就此略過(guò)倚喂。
問(wèn)題五:其它App播放聲音打斷
如果用戶(hù)當(dāng)時(shí)在后臺(tái)聽(tīng)音樂(lè)每篷,如QQ音樂(lè),或者喜馬拉雅這些App,這個(gè)時(shí)候播放視頻后焦读,其會(huì)被我們打斷子库,當(dāng)我們不再播放視頻的時(shí)候,自然需要繼續(xù)這些后臺(tái)聲音的播放矗晃。
首先仑嗅,我們需要先向設(shè)備注冊(cè)激活聲音打斷AudioSessionSetActive(YES);,當(dāng)然我們也可以通過(guò) [AVAudioSession sharedInstance].otherAudioPlaying;這個(gè)方法來(lái)判斷還有沒(méi)有其它業(yè)務(wù)的聲音在播放。 當(dāng)我們播放完視頻后喧兄,需要恢復(fù)其它業(yè)務(wù)或App的聲音无畔,這時(shí)我們可以在退到后臺(tái)的事件中調(diào)用如下方法:
- (void)applicationDidEnterBackground:(UIApplication *)application {
NSError *error =nil;
AVAudioSession *session = [AVAudioSession sharedInstance];
// [session setCategory:AVAudioSessionCategoryPlayback error:nil];
BOOL isSuccess = [session setActive:NO withOptions:AVAudioSessionSetActiveOptionNotifyOthersOnDeactivation error:&error];
if (!isSuccess) {
NSLog(@"__%@",error);
}else{
NSLog(@"成功了");
}
}
問(wèn)題六:在用戶(hù)插入和拔出耳機(jī)時(shí),導(dǎo)致視頻暫停吠冤,解決方法如下
//耳機(jī)插入和拔掉通知
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(audioRouteChangeListenerCallback:) name:AVAudioSessionRouteChangeNotification object:[AVAudioSession sharedInstance]];
//耳機(jī)插入浑彰、拔出事件
- (void)audioRouteChangeListenerCallback:(NSNotification*)notification {
NSDictionary *interuptionDict = notification.userInfo;
NSInteger routeChangeReason = [[interuptionDict valueForKey:AVAudioSessionRouteChangeReasonKey] integerValue];
switch (routeChangeReason) {
case AVAudioSessionRouteChangeReasonNewDeviceAvailable:
break;
case AVAudioSessionRouteChangeReasonOldDeviceUnavailable:
{
//判斷為耳機(jī)接口
AVAudioSessionRouteDescription *previousRoute =interuptionDict[AVAudioSessionRouteChangePreviousRouteKey];
AVAudioSessionPortDescription *previousOutput =previousRoute.outputs[0];
NSString *portType =previousOutput.portType;
if ([portType isEqualToString:AVAudioSessionPortHeadphones]) {
// 拔掉耳機(jī)繼續(xù)播放
if (self.playing) {
[self.player play];
}
}
}
break;
case AVAudioSessionRouteChangeReasonCategoryChange:
// called at start - also when other audio wants to play
break;
}
}
問(wèn)題七:打電話(huà)等中斷事件
//中斷的通知
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleInterruption:) name:AVAudioSessionInterruptionNotification object:[AVAudioSession sharedInstance]];
//中斷事件
- (void)handleInterruption:(NSNotification *)notification{
NSDictionary *info = notification.userInfo;
//一個(gè)中斷狀態(tài)類(lèi)型
AVAudioSessionInterruptionType type =[info[AVAudioSessionInterruptionTypeKey] integerValue];
//判斷開(kāi)始中斷還是中斷已經(jīng)結(jié)束
if (type == AVAudioSessionInterruptionTypeBegan) {
//停止播放
[self.player pause];
}else {
//如果中斷結(jié)束會(huì)附帶一個(gè)KEY值,表明是否應(yīng)該恢復(fù)音頻
AVAudioSessionInterruptionOptions options =[info[AVAudioSessionInterruptionOptionKey] integerValue];
if (options == AVAudioSessionInterruptionOptionShouldResume) {
//恢復(fù)播放
[self.player play];
}
}
}
小提示:如果不起作用拯辙,請(qǐng)檢查退到后臺(tái)事件中有什么其它的操作沒(méi)郭变。因?yàn)殡娫?huà)來(lái)時(shí),會(huì)調(diào)用退到后臺(tái)的事件涯保。
問(wèn)題七:內(nèi)存泄露問(wèn)題 當(dāng)我們釋放一個(gè)正在播放的視頻時(shí)诉濒,需要先調(diào)用pause方法,如果由于某些原因夕春,例如切前后臺(tái)時(shí)未荒,導(dǎo)致又調(diào)用了play方法,那么有可能會(huì)hold不住內(nèi)存空間而導(dǎo)致內(nèi)存泄漏及志。其實(shí)更靠譜的方法是片排,還要移除player加載的資源。
總的來(lái)說(shuō),AVPlayer能滿(mǎn)足一般的需求速侈,雖然坑不少率寡。最后,再來(lái)安利一下我自己封裝的LYAVPlayer,簡(jiǎn)單方便倚搬,支持cocoa pods冶共,只需幾行代碼即可完成播放:
LYAVPlayerView *playerView =[LYAVPlayerView alloc]init];
playerView.frame =CGRectMake(0, 64, ScreenWidth,200);
playerView.delegate =self;//設(shè)置代理
[self.view addSubview:playerView];
[playerView setURL:[NSURL URLWithString:VideoURL]];//設(shè)置播放的URL
[playerView play];//開(kāi)始播放
工程中pod 'LYAVPlayer','~> 1.0.1'
即可使用。有什么問(wèn)題請(qǐng)Issues我每界。