C#實現(xiàn)語音視頻錄制-附demo源碼
在很多語音視頻軟件系統(tǒng)中,經(jīng)常有將實時的音頻或視頻錄制為文件保存到磁盤的需求,比如,視頻監(jiān)控系統(tǒng)中錄制監(jiān)控到的視頻、視頻會議系統(tǒng)中錄制整個會議的過程、語音通話系統(tǒng)中錄制完整的對話內(nèi)容、等等?!?/p>
MFile組件(Oraycn.MFile.dll)是傲瑞實用組件之一,它可以將原始的語音數(shù)據(jù)和視頻數(shù)據(jù)按照指定的格式進行編碼,并將它們寫入到視頻文件(如.mp4)中。
一.緣起
最近正在做的一個網(wǎng)絡(luò)招聘平臺的項目,其中有一個模塊是這樣的,應(yīng)聘者可以通過該系統(tǒng)的客戶端錄制自己的視頻(自我介紹)上傳到服務(wù)器,而后,招聘者會在合適的時候瀏覽這些應(yīng)聘者的視頻。該模塊涉及到的主要技術(shù)就是語音視頻錄制技術(shù),它需要把從麥克風(fēng)采集到的語音數(shù)據(jù)和從攝像頭采集到的視頻數(shù)據(jù)編碼并寫到.mp4文件中。
要完成這些功能,具體來說,需要解決如下幾個技術(shù)問題:
(1)麥克風(fēng)數(shù)據(jù)采集
(2)攝像頭數(shù)據(jù)采集
(3)音頻數(shù)據(jù)編碼
(4)視頻數(shù)據(jù)編碼
(5)將編碼后的數(shù)據(jù)按.mp4文件格式寫入到文件容器中。
(6)保證音頻視頻播放的同步。
二.Demo實現(xiàn)
如果要從頭開始一步步解決這些問題,將是非常艱難的挑戰(zhàn)。幸運的是,我們可以通過已有組件的組合來實現(xiàn)這些功能,語音視頻數(shù)據(jù)的采集我們可以借助OMCS框架完成,后續(xù)的語音視頻編碼并生成mp4文件,我們可以借助MFile組件完成。為了更方便地講解,這里我們將給出一個具體的demo,它可以錄制從本地攝像頭和本地麥克風(fēng)采集的數(shù)據(jù)并生成mp4文件。demo運行的截圖如下所示:
接下來,我們來說說在這個demo中是如何一個個解決上述問題的。
1.語音數(shù)據(jù)采集
我們可以使用OMCS的MicrophoneConnector組件連接到自己的麥克風(fēng)設(shè)備,這樣,揚聲器就會播放采集到的語音,而且,我們可以通過通過IMultimediaManager暴露的AudioPlayed事件,來捕獲正在播放的語音數(shù)據(jù)。
2.視頻數(shù)據(jù)采集
同樣的,我們可以使用CameraConnector控件連接到自己的攝像頭設(shè)備,然后,定時器每隔100ms(假設(shè)幀頻為10fps)調(diào)用其GetCurrentImage方法獲得正在繪制的Bitmap。
3.后續(xù)步驟
后續(xù)的4步都可以交由MFile組件搞定,我們大概看一下MFile組件中VideoFileMaker類的簽名,就知道怎么做了:
MFile 結(jié)構(gòu)
對于使用者而言,MFile組件中的主要類的結(jié)構(gòu)圖如下所示:
其中,AudioFileMaker用于生成音頻文件、SilenceVideoFileMaker用于生成無聲的視頻文件、而VideoFileMaker用于生成既有聲音又有圖像的普通視頻文件。這三個類都從基類BaseMaker繼承,它們的使用方式也是一致的。接下來,我們僅僅詳細講解VideoFileMaker類的使用,SilenceVideoFileMaker 和 AudioFileMaker的使用方法可以類推之。
下面是VideoFileMaker類的public方法的簽名
代碼如下:
public class VideoFileMaker :IDisposable { /// <summary> /// 初始化視頻文件。 /// </summary> /// <param name="filePath">文件路徑</param> /// <param name="videoCodec">視頻編碼格式</param> /// <param name="videoWidth">視頻寬度</param> /// <param name="videoHeight">視頻高度</param> /// <param name="videoFrameRate">幀頻</param> /// <param name="audioCodec">音頻編碼格式</param> /// <param name="audioSampleRate">音頻采樣率?!咀ⅲ翰蓸游粩?shù)必須為16位】</param> /// <param name="audioChannelCount">聲道數(shù)</param> /// <param name="autoSyncToAudio">如果是實時錄制,則可傳入true,以音頻為基準進行同步。</param> void Initialize(string filePath, VideoCodecType videoCodec, int videoWidth, int videoHeight, int videoFrameRate, AudioCodecType audioCodec, int audioSampleRate, int audioChannelCount, bool autoSyncToAudio); /// <summary> /// 添加音頻幀。 /// </summary> void AddAudioFrame(byte[] audioframe); /// <summary> /// 添加視頻幀。如果autoSyncToAudio開啟,則自動同步到音頻。 /// </summary> void AddVideoFrame(Bitmap frame); /// <summary> /// 添加視頻幀。 /// </summary> /// <param name="frame">視頻幀</param> /// <param name="timeStamp">離開始時的時間長度</param> void AddVideoFrame(Bitmap frame, TimeSpan timeStamp); /// <summary> /// 關(guān)閉視頻文件。 /// </summary> /// <param name="waitFinished">如果還有幀等待寫入文件,是否等待它們?nèi)繉懭胛募?lt;/param> void Close(bool waitFinished); }
首先調(diào)用Initialize方法完成初始化,然后,循環(huán)調(diào)用AddAudioFrame和AddVideoFrame方法,當(dāng)完成視頻錄制時,則調(diào)用Close方法,即可。很簡單,不是嗎?
4.主要代碼
首先,我們以aa01用戶登錄到OMCS服務(wù)器,然后,在拖拽一個CameraConnector控件和一個MicrophoneConnector組件到主窗體上,然后,讓它們都連到自己的攝像頭和麥克風(fēng)。
this.multimediaManager = MultimediaManagerFactory.GetSingleton(); this.multimediaManager.Initialize("aa01", "", "127.0.0.1", 9900); this.cameraConnector1.BeginConnect("aa01"); this.microphoneConnector1.BeginConnect("aa01");
接下來,我們初始化VideoFileMaker組件:
this.videoFileMaker.Initialize("test.mp4", VideoCodecType.H264, this.multimediaManager.CameraVideoSize.Width, this.multimediaManager.CameraVideoSize.Height, 10, AudioCodecType.AAC, 16000, 1, true); this.timer = new System.Threading.Timer(new System.Threading.TimerCallback(this.Callback), null ,0, 100); this.multimediaManager.AudioPlayed += new ESBasic.CbGeneric<byte[]>(multimediaManager_AudioPlayed);
參數(shù)中設(shè)定,使用h.264對視頻進行編碼,使用aac對音頻進行編碼,并生成mp4格式的文件。然后,我們可以通過OMCS獲取實時的音頻數(shù)據(jù)和視頻數(shù)據(jù),并將它們寫到文件中。
void multimediaManager_AudioPlayed(byte[] audio) { this.videoFileMaker.AddAudioFrame(audio); } private void Callback(object state) { Bitmap bm = this.cameraConnector1.GetCurrentImage(); this.videoFileMaker.AddVideoFrame(bm); }
當(dāng)想結(jié)束錄制時,則調(diào)用Close方法:
this.videoFileMaker.Close(true);
這樣錄制生成的test.mp4文件就可以直接用我們的QQ影音或暴風(fēng)影音來播放了。
更多細節(jié),請查看demo源碼。
三.Demo下載
2014.11.26 現(xiàn)在錄制本地的語音、視頻、屏幕的最好的方案是MCapture + MFile,而不是通過OMCS繞一大圈,相應(yīng)的Demo源碼下載:Oraycn.RecordDemo.rar 。
當(dāng)然,如果是遠程錄制語音、視頻、屏幕,最好的方案是OMCS + MFile。
以上就是本文針對C#實現(xiàn)語音視頻錄制-附demo源碼的全部內(nèi)容,希望大家喜歡。