在很多语音视频软件系统中,经常有将实时的音频或视频录制为文件保存到磁盘的需求,比如,视频监控系统中录制监控到的视频、视频会议系统中录制整个会议的过程、语音通话系统中录制完整的对话内容、等等。
一.缘起
最近正在做的一个网络招聘平台的项目,其中有一个模块是这样的,应聘者可以通过该系统的客户端录制自己的视频(自我介绍)上传到服务器,而后,招聘者会在合适的时候浏览这些应聘者的视频。该模块涉及到的主要技术就是语音视频录制技术,它需要把从麦克风采集到的语音数据和从摄像头采集到的视频数据编码并写到.mp4文件中。要完成这些功能,具体来说,需要解决如下几个技术问题:
(1)麦克风数据采集
(2)摄像头数据采集
(3)音频数据编码
(4)视频数据编码
(5)将编码后的数据按.mp4文件格式写入到文件容器中。
(6)保证音频视频播放的同步。
二.Demo实现
如果要从头开始一步步解决这些问题,将是非常艰难的挑战。幸运的是,我们可以通过已有组件的组合来实现这些功能,语音视频数据的采集我们可以借助OMCS框架完成,后续的语音视频编码并生成mp4文件,我们可以借助MFile组件完成。为了更方便地讲解,这里我们将给出一个具体的demo,它可以录制从本地摄像头和本地麦克风采集的数据并生成mp4文件。demo运行的截图如下所示:
接下来,我们来说说在这个demo中是如何一个个解决上述问题的。
1.语音数据采集
我们可以使用OMCS的MicrophoneConnector组件连接到自己的麦克风设备,这样,扬声器就会播放采集到的语音,而且,我们可以通过通过IMultimediaManager暴露的AudioPlayed事件,来捕获正在播放的语音数据。
2.视频数据采集
同样的,我们可以使用CameraConnector控件连接到自己的摄像头设备,然后,定时器每隔100ms(假设帧频为10fps)调用其GetCurrentImage方法获得正在绘制的Bitmap。
3.后续步骤
后续的4步都可以交由MFile组件搞定,我们大概看一下MFile组件中VideoFileMaker类的签名,就知道怎么做了:
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">音频采样率。【注:采样位数必须为16位】</param>
/// <param name="audioChannelCount">声道数</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>
/// 关闭视频文件。
/// </summary>
/// <param name="waitFinished">如果还有帧等待写入文件,是否等待它们全部写入文件。</param>
void Close(bool waitFinished);
}
首先调用Initialize方法完成初始化,然后,循环调用AddAudioFrame和AddVideoFrame方法,当完成视频录制时,则调用Close方法,即可。很简单,不是吗?
4.主要代码
首先,我们以aa01用户登录到OMCS服务器,然后,在拖拽一个CameraConnector控件和一个MicrophoneConnector组件到主窗体上,然后,让它们都连到自己的摄像头和麦克风。
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);
参数中设定,使用h.264对视频进行编码,使用aac对音频进行编码,并生成mp4格式的文件。然后,我们可以通过OMCS获取实时的音频数据和视频数据,并将它们写到文件中。
void multimediaManager_AudioPlayed(byte[] audio)
{
this.videoFileMaker.AddAudioFrame(audio);
}
private void Callback(object state)
{
Bitmap bm = this.cameraConnector1.GetCurrentImage();
this.videoFileMaker.AddVideoFrame(bm);
}
当想结束录制时,则调用Close方法:
this.videoFileMaker.Close(true);
这样录制生成的test.mp4文件就可以直接用我们的QQ影音或暴风影音来播放了。
更多细节,请查看demo源码。
三.Demo下载
Demo源码:Oraycn.RecordDemo.rar
2014.11.26 现在录制本地的语音、视频、屏幕的最好的方案是MCapture + MFile,而不是通过OMCS绕一大圈,相应的Demo源码下载:Oraycn.RecordDemo.rar 。
当然,如果是远程录制语音、视频、屏幕,最好的方案是OMCS + MFile。
2015.6.18 整理全部相关demo如下:
(声卡/麦克风/摄像头/屏幕)采集&录制Demo:WinForm版本 、WPF版本。
声卡录制Demo、 混音&录制Demo、 同时录制(桌面+麦克风+声卡)Demo、 麦克风摄像头录制(可预览)、