这两天研究了FFmpeg获取DirectShow设备数据的方法,在此简单记录一下以作备忘。本文所述的方法主要是对应Windows平台的。
1. 列设备
ffmpeg -list_devices true -f dshow -i dummy
命令执行后输出的结果如下(注:中文的设备会出现乱码的情况)。列表显示设备的名称很重要,输入的时候都是使用“-f dshow -i video="{设备名}"”的方式。
我自己的机器上列出了以下设备:
[dshow @0388f5e0] DirectShow video devices
[dshow @0388f5e0] "Integrated Camera"
[dshow @0388f5e0] "screen-capture-recorder"
[dshow @0388f5e0] DirectShow audio devices
[dshow @0388f5e0] "鍐呰楹﹀厠椋?(Conexant20672 SmartAudi"
[dshow @0388f5e0] "virtual-audio-capturer"
下文的测试中,使用其中的两个视频输入:"Integrated Camera"和"screen-capture-recorder"。
注:音频设备出现乱码,这个问题的解决方法会随后提到。
2. 获取摄像头数据(保存为本地文件或者发送实时流)
2.1. 编码为H.264,保存为本地文件
下面这条命令,实现了从摄像头读取数据并编码为H.264,最后保存成mycamera.mkv。
ffmpeg -f dshow -i video="Integrated Camera" -vcodec libx264 mycamera.mkv
2.2. 直接播放摄像头的数据
使用ffplay可以直接播放摄像头的数据,命令如下:
ffplay -f dshow -i video="Integrated Camera"
如果设备名称正确的话,会直接打开本机的摄像头,如图所示。
注:除了使用DirectShow作为输入外,使用VFW也可以读取到摄像头的数据,例如下述命令可以播放摄像头数据:
ffplay -f vfwcap -i 0
此外,可以使用FFmpeg的list_options查看设备的选项:
ffmpeg -list_options true -f dshow -i video="Integrated Camera"
输出如下:
[dshow @ 03845420] DirectShow video device options
[dshow @ 03845420] Pin "鎹曡幏"
[dshow @ 03845420] pixel_format=bgr24 min s=640x480 fps=15 max s=640x480 fps=30
[dshow @ 03845420] pixel_format=bgr24 min s=640x360 fps=15 max s=640x360 fps=30
[dshow @ 03845420] pixel_format=bgr24 min s=352x288 fps=15 max s=352x288 fps=30
[dshow @ 03845420] pixel_format=bgr24 min s=320x240 fps=15 max s=320x240 fps=30
[dshow @ 03845420] pixel_format=bgr24 min s=800x448 fps=1 max s=800x448 fps=15
[dshow @ 03845420] pixel_format=bgr24 min s=960x544 fps=1 max s=960x544 fps=10
[dshow @ 03845420] pixel_format=bgr24 min s=1280x720 fps=1 max s=1280x720 fps=10
[dshow @ 03845420] pixel_format=bgr24 min s=424x240 fps=15 max s=424x240 fps=30
[dshow @ 03845420] pixel_format=yuyv422 min s=640x480 fps=15 max s=640x480 fps=30
[dshow @ 03845420] pixel_format=yuyv422 min s=640x360 fps=15 max s=640x360 fps=30
[dshow @ 03845420] pixel_format=yuyv422 min s=352x288 fps=15 max s=352x288 fps=30
[dshow @ 03845420] pixel_format=yuyv422 min s=320x240 fps=15 max s=320x240 fps=30
[dshow @ 03845420] pixel_format=yuyv422 min s=800x448 fps=1 max s=800x448 fps=15
[dshow @ 03845420] pixel_format=yuyv422 min s=960x544 fps=1 max s=960x544 fps=10
[dshow @ 03845420] pixel_format=yuyv422 min s=1280x720 fps=1 max s=1280x720 fps=10
[dshow @ 03845420] pixel_format=yuyv422 min s=424x240 fps=15 max s=424x240 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=640x480 fps=15 max s=640x480 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=640x360 fps=15 max s=640x360 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=352x288 fps=15 max s=352x288 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=320x240 fps=15 max s=320x240 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=800x448 fps=15 max s=800x448 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=960x544 fps=15 max s=960x544 fps=30
[dshow @ 03845420] vcodec=mjpeg min s=1280x720 fps=15 max s=1280x720 fps=30
可以通过输出信息设置摄像头的参数。
例如,设置摄像头分辨率为1280x720
ffplay -s 1280x720 -f dshow -i video="Integrated Camera"
设置分辨率为424x240
ffplay -s 424x240 -f dshow -i video="Integrated Camera"
2.3. 编码为H.264,发布UDP
下面这条命令,实现了:获取摄像头数据->编码为H.264->封装为UDP并发送至组播地址。
ffmpeg -f dshow -i video="Integrated Camera" -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -f h264 udp://233.233.233.223:6666
注1:考虑到提高libx264的编码速度,添加了-preset:v ultrafast和-tune:v zerolatency两个选项。
注2:高分辨率的情况下,使用UDP可能出现丢包的情况。为了避免这种情况,可以添加–s 参数(例如-s 320x240)调小分辨率。
2.4. 编码为H.264,发布RTP
下面这条命令,实现了:获取摄像头数据->编码为H.264->封装为RTP并发送至组播地址。
ffmpeg -f dshow -i video="Integrated Camera" -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -f rtp rtp://233.233.233.223:6666>test.sdp
注1:考虑到提高libx264的编码速度,添加了-preset:v ultrafast和-tune:v zerolatency两个选项。
注2:结尾添加“>test.sdp”可以在发布的同时生成sdp文件。该文件可以用于该视频流的播放。
2.5. 编码为H.264,发布RTMP
下面这条命令,实现了:获取摄像头数据->编码为H.264->并发送至RTMP服务器。
ffmpeg -f dshow -i video="Integrated Camera" -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -f flv rtmp://localhost/oflaDemo/livestream
2.6. 编码为MPEG2,发布UDP
与编码为H.264类似,指明-vcodec即可。
ffmpeg -f dshow -i video="Integrated Camera" -vcodec mpeg2video -f mpeg2video udp://233.233.233.223:6666
播放MPEG2的UDP流如下。指明-vcodec为mpeg2video即可
ffplay -vcodec mpeg2video udp://233.233.233.223:6666
3. 屏幕录制(Windows平台下保存为本地文件或者发送实时流)
Linux下使用FFmpeg进行屏幕录制相对比较方便,可以使用x11grab,使用如下的命令:
ffmpeg -f x11grab -s 1600x900 -r 50 -vcodec libx264 –preset:v ultrafast –tune:v zerolatency -crf 18 -f mpegts udp://localhost:1234
详细时使用方式可以参考这篇文章:DesktopStreaming With FFmpeg for Lower Latency
Linux录屏在这里不再赘述。在Windows平台下屏幕录像则要稍微复杂一些。在Windows平台下,使用-dshow取代x11grab。一句话介绍:注册录屏dshow滤镜(例如screen-capture-recorder),然后通过dshow获取录屏图像然后编码处理。
因此,在使用FFmpeg屏幕录像之前,需要先安装dshow滤镜。在这里推荐一个软件:screen capture recorder。安装这个软件之后,就可以通过FFmpeg屏幕录像了。
screen capture recorder项目主页:
http://sourceforge.net/projects/screencapturer/
下载地址:
http://sourceforge.net/projects/screencapturer/files
下载完后,一路“Next”即可安装完毕。注意,需要Java运行环境(Java Runtime Environment),如果没有的话下载一个就行。
screen capture recorder本身就可以录屏,不过这里我们使用FFmpeg进行录屏。
3.1. 编码为H.264,保存为本地文件
下面的命令可以将屏幕录制后编码为H.264并保存为本地文件。
ffmpeg -f dshow -i video="screen-capture-recorder" -r 5 -vcodec libx264 -preset:v ultrafast -tune:v zerolatency MyDesktop.mkv
注:“-r 5”的意思是把帧率设置成5。
最后得到的效果如下图。
此外,也可以录声音,声音输入可以分成两种:一种是真人说话的声音,通过话筒输入;一种是虚拟的声音,即录屏的时候电脑耳机里的声音。下面两条命令可以分别录制话筒的声音和电脑耳机里的声音。
录屏,伴随话筒输入的声音
ffmpeg -f dshow -i video="screen-capture-recorder" -f dshow -i audio="鍐呰楹﹀厠椋?(Conexant 20672 SmartAudi" -r 5 -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -acodec libmp3lame MyDesktop.mkv
上述命令有问题:audio那里有乱码,把乱码ANSI转UTF-8之后,开始测试不行,后来发现是自己疏忽大意,乱码部分转码后为“内装麦克风 ”,然后接可以正常使用了。因此,命令应该如下图所示:
ffmpeg -f dshow -i video="screen-capture-recorder" -f dshow -i audio="内装麦克风 (Conexant 20672 SmartAudi" -r 5 -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -acodec libmp3lame MyDesktop.mkv
注:
如果不熟悉ANSI转码UTF-8的话,还有一种更简单的方式查看设备的名称。即不使用FFmpeg查看系统DirectShow输入设备的名称,而使用DirectShow SDK自带的工具GraphEdit(或者网上下一个GraphStudioNext)查看输入名称。
打开GraphEdit选择“图像->插入滤镜”
然后就可以通过查看Audio Capture Sources来查看音频输入设备的简体中文名称了。从图中可以看出是“内装麦克风 (Conexant 20672 SmartAudi”。
PS:感觉这条命令适合做讲座之类的时候使用
录屏,伴随耳机输入的声音
ffmpeg -f dshow -i video="screen-capture-recorder" -f dshow -i audio="virtual-audio-capturer" -r 5 -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -acodec libmp3lame MyDesktop.mkv
PS:测这条命令的时候,这在听歌,因此录制的视频中的音频就是那首歌曲。
3.2. 编码为H.264,发布UDP
下面的命令可以将屏幕录制后编码为H.264并封装成UDP发送到组播地址
ffmpeg -f dshow -i video="screen-capture-recorder" -r 5 -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -f h264 udp://233.233.233.223:6666
注1:考虑到提高libx264的编码速度,添加了-preset:v ultrafast和-tune:v zerolatency两个选项。
注2:高分辨率的情况下,使用UDP可能出现丢包的情况。为了避免这种情况,可以添加–s 参数(例如-s 320x240)调小分辨率。
3.3. 编码为H.264,发布RTP
下面的命令可以将屏幕录制后编码为H.264并封装成RTP并发送到组播地址
ffmpeg -f dshow -i video="screen-capture-recorder" -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -f rtp rtp://233.233.233.223:6666>test.sdp
注1:考虑到提高libx264的编码速度,添加了-preset:v ultrafast和-tune:v zerolatency两个选项。
注2:结尾添加“>test.sdp”可以在发布的同时生成sdp文件。该文件可以用于该视频流的播放。如下命令即可播放:
3.4. 编码为H.264,发布RTMP
原理同上,不再赘述。
ffmpeg -f dshow -i video="Integrated Camera" -vcodec libx264 -preset:v ultrafast -tune:v zerolatency -f flv rtmp://localhost/oflaDemo/livestream
注意:播放RTMP的时候,-max_delay参数会比较明显的影响延迟,将此参数值设定小一些,有利于降低延时。
ffplay -max_delay 100000 "rtmp://localhost/oflaDemo/livestream live=1"
4.另一种屏幕录制的方式(2014.10.1更新)
最近发现FFmpeg还有一个专门用于Windows下屏幕录制的设备:gdigrab。
gdigrab是基于GDI的抓屏设备,可以用于抓取屏幕的特定区域。在这里记录一下gdigrab的用法。
gdigrab通过设定不同的输入URL,支持两种方式的屏幕抓取:
(1)“desktop”:抓取整张桌面。或者抓取桌面中的一个特定的区域。
(2)“title={窗口名称}”:抓取屏幕中特定的一个窗口。
下面举几个例子。
最简单的抓屏:
ffmpeg -f gdigrab -i desktop out.mpg
从屏幕的(10,20)点处开始,抓取640x480的屏幕,设定帧率为5
ffmpeg -f gdigrab -framerate 5 -offset_x 10 -offset_y 20 -video_size 640x480 -i desktop out.mpg
分享到:
相关推荐
这里是利用FFmpeg获取网络摄像头的数据,然后解码播放。 开发环境:win7+opencv3.0+ffmpeg+VS2013
这是windows下ffmpeg推流桌面与摄像头数据到流媒体服务器源码,该软件里推流和视频保存使用FFMPEG库完成,视频和音频可以同步推流和录制,FFMPEG本身支持跨平台编译开发,QT也支持跨平台,在Android、Linux、windows...
之前一直用Directshow技术采集摄像头数据,但是觉得涉及的细节比较多,要开发者比较了解Directshow的框架知识,学习起来有一点点难度。最近发现很多人问怎么用FFmpeg采集摄像头图像,事实上FFmpeg很早就支持通过...
1.开发环境:vs2015; 2.FFMPEG获取本地摄像头视频流,OPENCV嵌入MFC对话框中显示; 3.程序包中包括FFMPEG和OPENCV资源包; 4.环境已配置好,拿到代码修改查找摄像头名称即可编译运行;
使用qt编写的录屏和录制摄像头,实现原理是执行ffmpeg命令行来实现,具体可以看我的博客
ffmpeg录屏并通过UDP发送出去,接收端的可以直接播放或者保存: 1)播放:ffplay -f h264 udp://本机IP:6666 2)保存:ffmpeg -i udp://本机IP:6666 -c copy dump.flv 由于是采用UDP方式发送出去的,接收端是否在线...
基于Qt和FFmpeg开发的一款简易录屏软件源码+项目说明.zip 基于Qt、FFmpeg开发的一款简易录屏软件 1、支持屏幕录制和音频录制 2、支持本地录制和网络推流 3、推流服务器使用Nginx-RTMP推流模块 【备注】 1、该...
Qt+FFmpeg简单实现录屏并保存为MP4视频,可执行程序,非源码。 https://blog.csdn.net/qq_41632571/article/details/130267695
FFmpeg深度插件全面支持电脑、安卓和苹果手机,不是那种用ffmpeg.exe载入线程的,那种只支持电脑,而我这个是全面支持PC(Windows)、Android和IOS系统,并且优化了图像和码率,效果非常好,亲测完美兼容UNITY5.X到...
Windows平台,Qt调用FFmpeg.exe进行录屏。支持操作,开始录屏-暂停-结束录屏。
实现录屏+录音,发现bug修改,音csdn资源不能修改,所以重传
功能:支持选择码率、推流摄像头或者桌面画面到流媒体服务器、或者是将摄像头、桌面画面录制保存到本地。 Qt版本: 5.12.6 FFMPEG版本: 4.2.2 特点:采用Qt、ffmpeg 开发,功能完善,可以很方便的移植到其他系统。
使用 ffmpeg dshow 实现系统录屏 .Net(C#) 帮助类。 需要安装 Setup Screen Capturer Recorder
利用ffmpeg的API从USB摄像头获取视频并保存为H264的TS流的C语言源代码 利用ffmpeg的API从USB摄像头获取视频并保存为H264的TS流的C语言源代码
利用ffmpeg编解码库推本地图片或者本地摄像头成rtmp流,资源内容为cpp文件,ffmpeg编译时需要注意版本,不然会出现未定义出错,ffmpeg新旧库更新问题。 可更改代码中部分参数,降低延迟以及提高推流图像的质量。
基于ffmpeg的API,从摄像头采集数据并将其保存为mp4或者avi文件。有详细的中文备注,测试可用。-Ffmpeg' s API to collect data from the camera and save it as mp4 or avi file. Detailed Chinese notes, tests ...
采用内存dc抓取屏幕图像并送入ffmpeg压制为avi视频
Linux系统下使用ffmpeg录屏的代码,使用C++语言编写。
王纲老师讲的FFMPEG调取摄像头录像,很好的学习参考资料。
ffmpeg sdl2.0 播放usb摄像头 开发环境vs2010