https://www.jianshu.com/p/714ce954e628css
最近接手公司的直播項目,對之前遺留的問題作處理和優化, 因而順便看了下阿里雲直播的文檔,在下面寫下對直播的理解和遇到的問題, 阿里雲售後特別好,一對一解決問題速度很快,若是遇到解決不了的問題能夠發工單提問,效率很高產品->視頻直播->文檔&SDK->聯繫客服->工單支持而後選擇本身遇到問題的產品類型提交工單便可,通常兩個小時內能夠獲得回覆
html
工欲善其事必先利其器,先作準備工做
Step1. 訪問 阿里雲官網,點左上角 登陸
Step2. 登陸視頻直播控制檯
在 視頻直播服務產品主頁登陸控制檯。控制檯會檢查所依賴服務的開通狀態,請按頁面引導操做。
Step3. 在 域名管理 中,新建域名。直播域名須要進行備案審覈,審覈經過後便可使用,未備案的域名請先進行備案,
備案流程
Step4. CNAME綁定將您添加的直播域名的DNS CNAME紀錄修改成直播域名管理詳情頁面上顯示的CNAME綁定地址。咱們須要把阿里雲提供的推流地址和直播域名進行綁定,這樣當推流到直播域名時會推流到咱們的直播中心。
Step4. 獲取推流和播放地址在 域名管理 中,點擊直播加速域名 管理 :獲取推流和直播地址
Step5. 鑑權配置直播流媒體的推送和播放採用同一套鑑權方案,能夠在控制檯的鑑權配置中進行配置,詳細瞭解
鑑權配置。瀏覽器
推流
Step1. 獲取鑑權後的推流地址:
直播控制檯 - 域名管理 - 直播域名管理詳情頁 - 基本信息 取得推流地址以下: rtmp://video-center.alivecdn.com/AppName/StreamName?vhost=live.aliyun.com 使用直播控制檯 - 域名管理 - 直播域名管理詳情頁 - 鑑權配置 頁面的鑑權URL計算器計算鑑權URL: 輸入推流地址(AppName、StreamName可自行修改)、鑑權KEY、有效時間,點擊<生成>按鈕便可獲得鑑權URL。
Step2. 推流操做推流地址:rtmp://video-center.alivecdn.com/APPName/StreamName?vhost=live.aliyun.com
服務器
如下面的推流地址爲例,參數設置爲:FMS URL / URL: rtmp://video-center.alivecdn.com/AppName 播放路徑/串碼流(若是存在)/ 流祕鑰: StreamName?vhost=live.aliyn.com 如您開啓了鑑權,則鑑權參數也一併放在 Mac版obs的流密鑰與Windows版播放路徑/串碼流(若是存在)中。
播放
客戶能夠根據實際業務場景靈活搭配使用,須要在移動端瀏覽器、移動H5端進行播放,建議使用HLS(M3U8)方式進行播放,無需集成SDK;非移動端或者已集成SDK的,低併發量並須要有更小的延時,可以使用RTMP,高併發量建議使用FLV。
Step1. Web頁面後臺直接預覽使用OBS等工具使用鑑權URL推流後,可在 直播控制檯 - 流管理 - 正在推流 頁面查詢到正在直播的推流記錄,經過 直播地址 可查詢播放地址,並可預覽播放。
Step2. 經過VLC預覽
下載VLC默認安裝後無需作額外設置,文件—>打開網絡串流,填寫播放地址並點擊打開後開始播放。
網絡
iOS 推流SDK開發包
iOS推流使用說明session
目前SDK的橫屏推流須要再推流界面的Controller中將手機豎屏鎖定(只容許Portrait一個方向),若是須要橫豎屏切換,須要對UI作一套橫豎屏的適配,監控手機的狀態並做出相應的佈局
(下面是推流,直播界面的代碼和設置)
註冊通知,根據不一樣狀況作不一樣判斷
併發
[[UIDevice currentDevice] beginGeneratingDeviceOrientationNotifications]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleDeviceOrientationDidChange:) name:UIDeviceOrientationDidChangeNotification object:nil];//監控手機感應狀態 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appResignActive) name:UIApplicationWillResignActiveNotification object:nil];//退入後臺中止推流 由於iOS後臺機制,不能知足充分的攝像頭採集和GPU渲染 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appBecomeActive) name:UIApplicationDidBecomeActiveNotification object:nil];// 回到前臺從新推流``` ` #pragma mark - 推流Session 建立 銷燬`
(void)createSession{app
AlivcLConfiguration *configuration = [[AlivcLConfiguration alloc] init];
configuration.url = @"rtmp://video-center.alivecdn.com/3b42277fcb90445096dd72d4a11ef420/";
configuration.videoMaxBitRate = 1500 * 1000;
configuration.videoBitRate = 600 * 1000;
configuration.videoMinBitRate = 400 * 1000;
configuration.audioBitRate = 64 * 1000;
configuration.videoSize = CGSizeMake(360, 640);// 橫屏狀態寬高不須要互換
configuration.fps = 20;
configuration.preset = AVCaptureSessionPresetiFrame1280x720;
configuration.screenOrientation =
// AlivcLiveScreenVertical = 0,
// AlivcLiveScreenHorizontal = 1,;
// 重連時長
configuration.reconnectTimeout = 5;
// 水印
configuration.waterMaskImage = [UIImage imageNamed:@"watermask"];
configuration.waterMaskLocation = 1;
configuration.waterMaskMarginX = 10;
configuration.waterMaskMarginY = 10;
// 攝像頭方向
if (self.currentPosition) {
configuration.position = self.currentPosition;
// AVCaptureDevicePositionBack 後置
// AVCaptureDevicePositionFront 前置
} else {
configuration.position = AVCaptureDevicePositionFront;
self.currentPosition = AVCaptureDevicePositionFront;
}
configuration.frontMirror = YES;async
// alloc session
self.liveSession = [[AlivcLiveSession alloc] initWithConfiguration:configuration];
self.liveSession.delegate = self;
// 是否靜音推流
self.liveSession.enableMute = self.muteButton.selected;
// 開始預覽
[self.liveSession alivcLiveVideoStartPreview];
// 開始推流
[self.liveSession alivcLiveVideoConnectServer];ide
NSLog(@"開始推流");
dispatch_async(dispatch_get_main_queue(), ^{
// 預覽view
[self.view insertSubview:[self.liveSession previewView] atIndex:0];
});
self.exposureValue = 0;
}
(void)destroySession{
[self.liveSession alivcLiveVideoDisconnectServer];
[self.liveSession alivcLiveVideoStopPreview];
[self.liveSession.previewView removeFromSuperview];
self.liveSession = nil;
NSLog(@"銷燬推流");
}```#pragma mark - Notification通知響應
- (void)appResignActive{ // 退入後臺中止推流 由於iOS後臺機制,不能知足充分的攝像頭採集和GPU渲染 [self destroySession]; // 監聽電話 _callCenter = [[CTCallCenter alloc] init]; _isCTCallStateDisconnected = NO; _callCenter.callEventHandler = ^(CTCall* call) { if ([call.callState isEqualToString:CTCallStateDisconnected]) { _isCTCallStateDisconnected = YES; } else if([call.callState isEqualToString:CTCallStateConnected]) { _callCenter = nil; } }; NSLog(@"退入後臺"); } - (void)appBecomeActive{ if (_isCTCallStateDisconnected) { sleep(2); } // 回到前臺從新推流 [self createSession]; NSLog(@"回到前臺"); } - (void)handleDeviceOrientationDidChange:(UIInterfaceOrientation)interfaceOrientation { UIDevice *device = [UIDevice currentDevice] ; switch (device.orientation) { case UIDeviceOrientationFaceUp: NSLog(@"屏幕朝上平躺"); break; case UIDeviceOrientationFaceDown: NSLog(@"屏幕朝下平躺"); break; case UIDeviceOrientationUnknown: NSLog(@"未知方向"); break; case UIDeviceOrientationLandscapeLeft: { NSLog(@"屏幕向左橫置"); [self destroySession];//摧毀推流 _isScreenHorizontal = YES; // 橫屏 YES [self createSession]; //從新推流 } break; case UIDeviceOrientationLandscapeRight: NSLog(@"屏幕向右橫置"); break; case UIDeviceOrientationPortrait: { NSLog(@"屏幕直立"); [self destroySession]; // 摧毀推流 _isScreenHorizontal = NO; //豎屏爲NO [self createSession]; // 從新推流 } break; case UIDeviceOrientationPortraitUpsideDown: NSLog(@" 屏幕直立 上下顛倒"); break; default: NSLog(@"沒法辨認"); break; } }``` `//推流代理,根據實際狀況作出反應`
(void)alivcLiveVideoLiveSession:(AlivcLiveSession *)session error:(NSError *)error{
dispatch_async(dispatch_get_main_queue(), ^{
NSString *msg = [NSString stringWithFormat:@"%zd %@",error.code, error.localizedDescription];
UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Live Error" message:msg delegate:nil cancelButtonTitle:@"取消" otherButtonTitles:@"從新鏈接", nil];
alertView.delegate = self;
[alertView show];
});
NSLog(@"liveSession Error : %@", error);
}
(void)alivcLiveVideoLiveSessionNetworkSlow:(AlivcLiveSession *)session {
UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"提示" message:@"當前網絡環境較差" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
[alertView show];
self.textView.text = @"網速過慢,影響推流效果,拉流端會形成卡頓等,建議暫停直播";
NSLog(@"網速過慢");
}
(void)alivcLiveVideoLiveSessionConnectSuccess:(AlivcLiveSession *)session {
NSLog(@"推流 connect success!");
}
(void)alivcLiveVideoReconnectTimeout:(AlivcLiveSession *)session error:(NSError *)error {
dispatch_async(dispatch_get_main_queue(), ^{
UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"提示" message:[NSString stringWithFormat:@"重連超時-error:%ld", error.code] delegate:nil cancelButtonTitle:@"OK" otherButtonTitles: nil];
[alertView show];
});
NSLog(@"重連超時");
}
(void)alivcLiveVideoOpenAudioSuccess:(AlivcLiveSession *)session {
// dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"YES" message:@"麥克風打開成功" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
// });
}
(void)alivcLiveVideoOpenVideoSuccess:(AlivcLiveSession *)session {
// dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"YES" message:@"攝像頭打開成功" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
// });
}
(void)alivcLiveVideoLiveSession:(AlivcLiveSession *)session openAudioError:(NSError *)error {
dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Error" message:@"麥克風獲取失敗" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
});
}
(void)alivcLiveVideoLiveSession:(AlivcLiveSession *)session openVideoError:(NSError *)error {
// dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Error" message:@"攝像頭獲取失敗" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
// });
}
(void)alivcLiveVideoLiveSession:(AlivcLiveSession *)session encodeAudioError:(NSError *)error {
// dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Error" message:@"音頻編碼初始化失敗" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
// });
}
(void)alivcLiveVideoLiveSession:(AlivcLiveSession *)session encodeVideoError:(NSError *)error {
// dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Error" message:@"視頻編碼初始化失敗" delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
// });
}
(void)alivcLiveVideoLiveSession:(AlivcLiveSession *)session bitrateStatusChange:(ALIVC_LIVE_BITRATE_STATUS)bitrateStatus {
// dispatch_async(dispatch_get_main_queue(), ^{
// UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"YES" message:[NSString stringWithFormat:@"ALIVC_LIVE_BITRATE_STATUS = %ld", bitrateStatus] delegate:nil cancelButtonTitle:@"肯定" otherButtonTitles: nil];
// [alertView show];
// });
NSLog(@"碼率變化 %ld", bitrateStatus);
}
(void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
if (buttonIndex != alertView.cancelButtonIndex) {
[self.liveSession alivcLiveVideoConnectServer];
} else {
[self.liveSession alivcLiveVideoDisconnectServer];
}
}
> 視頻播放
>> AlivcMediaPlayer
`iOS媒體播放器SDK是在iOS平臺上使用的軟件開發工具包(Software Developement Kit),爲iOS開發者提供簡單易用的接口,幫助開發者實現iOS平臺上的媒體播放應用開發。該SDK對目前主流的視頻格式都提供了良好的支持,支持本地和網絡媒體的播放,彌補了系統播放器在媒體格式上的不足。(記得添加AliyunPlayerSDK.framework)
若是須要適配橫豎屏要對UI作適配,只作橫屏的話能夠將頁面旋轉90度便可 ` ###### 播放器通知定義: AliVcMediaPlayerLoadDidPreparedNotification:播放器初始化視頻文件完成通知,調用prepareToPlay函數,會發送該通知,表明視頻文件已經準備完成,此時能夠在這個通知中獲取到視頻的相關信息,如視頻分辨率,視頻時長等。 AliVcMediaPlayerPlaybackDidFinishNotification:播放完成通知。當視頻播放完成後會收到此通知。播放完成會有幾種狀況, 1. 當用戶調用stop後視頻結束完成。 2. 視頻正常播放結束。 ` AliVcMediaPlayerStartCachingNotification:播放器開始緩衝視頻時發送該通知,當播放網絡文件時,網絡狀態不佳或者調用seekTo時,此通知告訴用戶網絡下載數據已經開始緩衝。 AliVcMediaPlayerEndCachingNotification:播放器結束緩衝視頻時發送該通知,當數據已經緩衝完,告訴用戶已經緩衝結束,來更新相關UI顯示。 AliVcMediaPlayerPlaybackErrorNotification:播放器播放失敗發送該通知,並在該通知中能夠獲取到錯誤碼。 AliVcMediaPlayerSeekingDidFinishNotification:播放器位置改變完成後發送該通知。 AliVcMediaPlayerFirstFrameNotification:播放器狀態首幀顯示後發送的通知。` 播放器通知發送邏輯: 1. 調用prepareToPlay成功後發送AliVcMediaPlayerLoadDidPreparedNotification通知,失敗則會發送AliVcMediaPlayerPlaybackErrorNotification。 2. 調用play、pause、stop、prepareToPlay、seekTo失敗後發送AliVcMediaPlayerPlaybackErrorNotification通知。 3. 調用stop/reset成功後播放視頻結束髮送AliVcMediaPlayerPlaybackDidFinishNotification通知,播放器自動播放結束也會發送AliVcMediaPlayerPlaybackDidFinishNotification通知。 4. 調用seekTo成功後發送AliVcMediaPlayerSeekingDidFinishNotification通知。 5. AliVcMediaPlayerStartCachingNotification和AliVcMediaPlayerEndCachingNotification通知,這個是在網絡視頻緩衝數據不足以夠播放後會發送此通知,通常網絡視頻在調用seekTo後會發送此通知。` ` //初始化播放器的類`
-(void) playVideo
{
player = [[AliVcMediaPlayer alloc] init];
//建立播放器,傳入顯示窗口
/**