EmguCV学习笔记 VB.Net 9.1 VideoCapture类

版权声明:本文为博主原创文章,转载请在显著位置标明本文出处以及作者网名,未经作者允许不得用于商业目的。

EmguCV是一个基于OpenCV的开源免费的跨平台计算机视觉库,它向C#和VB.NET开发者提供了OpenCV库的大部分功能。

教程VB.net版本请访问:EmguCV学习笔记 VB.Net 目录-CSDN博客

教程C#版本请访问:EmguCV学习笔记 C# 目录-CSDN博客

笔者的博客网址:https://blog.csdn.net/uruseibest

教程配套文件及相关说明以及如何获得pdf教程和代码,请移步:EmguCV学习笔记

学习VB.Net知识,请移步: vb.net 教程 目录_vb中如何用datagridview-CSDN博客

 学习C#知识,请移步:C# 教程 目录_c#教程目录-CSDN博客

9.1 VideoCapture

VideoCapture类是用于处理视频输入的类,它提供了多种方法用于从摄像头、视频文件、网络流等不同的源读取视频数据。

9.1.1 构造函数

VideoCapture类提供了2个构造函数:

1、Public Sub New (Optional camIndex As Integer = 0, Optional captureApi As VideoCapture. API = VideoCapture.API.Any)

  1. camIndex:要打开的摄像头的编号,例如0表示打开第一个摄像头
  2. captureApi:指定videocapture对象使用的视频捕获API,这是一个VideoCapture.API枚举,Windows下常用的成员有:
    1. Any:自动选择最佳的视频捕获API
    2. DShow:使用DirectShow视频捕获API。
    3. Vfw:使用Video for Windows视频捕获API。
    4. Ffmpeg:使用FFmpeg视频捕获API,但是需要安装相应的FFmpeg库。

2、Public Sub New (fileName As String, Optional captureApi As VideoCapture. API = VideoCapture.API.Any)

  1. fileName:要打开的视频文件路径或网络流地址。
  2. captureApi:指定videocapture对象使用的视频捕获API。

由于这个构造函数的两个参数均有默认值,所以在实际使用中,可以使用以下方法:

Public Sub New ()

Public Sub New (camIndex As Integer)

Public Sub New (fileName As String)

通常可以采用以下方法来从摄像头或文件获得视频:

1、打开本地第一个摄像头,

Dim capture As New VideoCapture(0)

2、打开本地视频文件

Dim capture As New VideoCapture ("D:\video.mp4")

3、打开网络流

Dim capture As New VideoCapture ("http://example.com/video.m3u8")

在EmguCV中,VideoCapture类支持许多网络视频协议,包括RTSP、HTTP、FTP、MMS等。通过使用不同的网络视频协议,可以从远程摄像头、网络摄像头或者网络视频流中获取视频数据。下面分别介绍一下这几种网络视频协议的使用方法。

1. RTSP协议

RTSP协议是一种用于实时数据传输的协议,常用于远程视频监控。

Dim capture As New VideoCapture("rtsp://192.168.1.100:554/live.sdp") ' 打开RTSP视频流

Dim capture As New VideoCapture ("rtsp://admin:***@ 192.168.1.100:554");

其中的网址是RTSP视频流的URL地址,需要根据实际修改。

2. HTTP协议

HTTP协议是一种广泛应用于互联网的协议,可以通过HTTP视频流获取网络摄像头或者网络视频流的视频数据。例如:

Dim capture As New VideoCapture("http://192.168.1.100:8080/video")

Dim capture As New VideoCapture("http://username:pass@cam_address/video.cgi? ")

其中的网址是HTTP视频流的URL地址,需要根据实际修改。

3. FTP协议

FTP协议是一种文件传输协议,可以通过FTP视频流获取网络摄像头或者网络视频流的视频数据。

Dim capture As New VideoCapture("ftp://192.168.1.100:21/video.mp4")

其中网址是FTP视频流的URL地址,需要根据实际修改。

4. MMS协议

MMS协议是一种多媒体流传输协议,可以通过MMS视频流获取网络摄像头或者网络视频流的视频数据。

Dim capture As New VideoCapture("mms://192.168.1.100:1755/video.wmv")

其中网址是MMS视频流的URL地址,需要根据实际修改。

注意:并非所有的网络视频协议都适用于所有的视频输入设备,有些视频输入设备可能不支持某些协议。在使用VideoCapture类打开网络视频流时,需要确认网络视频协议是否被支持。可以通过使用VideoCapture的BackendName属性获取VideoCapture类的后端名称,以判断是否支持某种网络视频协议。

【代码位置:frmChapter9_1】Button1_Click

    Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click

        '下面测试视频的网址来自https://rtsp.stream/ ,可以用以下方法获得

        '打开https://rtsp.stream/,点击Free下的"Get started"填写邮箱,打开邮箱,打开指定的链接后,会给出2个免费的有效地址。

        Dim vc As New VideoCapture("rtsp://zephyr.rtsp.stream/pattern?streamKey=3c1b3e15c8904e1b7eea13468a085b3d")

        Dim backendName As String = vc.BackendName

        If backendName = "FFMPEG" Then ' 判断是否支持FFMPEG

            '如果支持FFMPEG,可以使用RTSPHTTPFTPMMS等网络视频协议

        End If

End Sub

9.1.2 Get和Set方法

Get和Set方法用于获取或设置视频输入源的相关参数,例如视频分辨率、帧率、亮度、对比度等。这两个方法的重要参数是一个CapProp枚举,它包含很多重要的成员。例如:

1、获得VideoCapture对象视频相关属性:

VideoCapture.Get(CapProp.Fps) ' 获取视频帧率

VideoCapture.Get(CapProp.FrameWidth) ' 获取视频宽度

VideoCapture.Get(CapProp.FrameHeight) ' 获取视频高度

VideoCapture.Get(CapProp.FrameCount) ' 获取视频总帧数

VideoCapture.Get(CapProp.PosFrames) ' 获取视频当前帧数

VideoCapture.Get(CapProp.PosAvti) '获取视频当前帧的位置的属性值。

2、设置VideoCapture对象视频相关属性:

VideoCapture.Set(CapProp.FrameWidth, 640) ' 设置视频宽度为640像素

VideoCapture.Set(CapProp. FrameHeight,, 480) ' 设置视频高度为480像素

VideoCapture.Set(CapProp. Fps,, 30) ' 设置视频帧率为30帧/秒

VideoCapture.Set(CapProp. Brightness, 50) ' 设置视频亮度为50

VideoCapture.Set(CapProp. Contrast, 50) ' 设置视频对比度为50

【代码位置:frmChapter9_1】Button2_Click

    '获得视频信息

Private Sub Button2_Click(sender As Object, e As EventArgs) Handles Button2.Click

        Dim vc As New VideoCapture("C:\learnEmgucv\movie1.mov")

        Dim width As Double = vc.Get(CapProp.FrameWidth)

        Dim height As Double = vc.Get(CapProp.FrameHeight)

        Dim fps As Double = vc.Get(CapProp.Fps)

        Dim FrameCount As Double = vc.Get(CapProp.FrameCount)

        Dim Brightness As Double = vc.Get(CapProp.Brightness)

        Dim Contrast As Double = vc.Get(CapProp.Contrast)

        Console.WriteLine("视频宽度:" & width)

        Console.WriteLine("视频高度:" & height)

        Console.WriteLine("视频帧率:" & fps)

        Console.WriteLine("视频总帧数:" & FrameCount)

        Console.WriteLine("视频亮度:" & Brightness)

        Console.WriteLine("视频对比度:" & Contrast)

        vc.Dispose()

End Sub

注意:并非所有的视频属性都适用于所有的视频输入设备,有些视频输入设备可能不支持某些属性,比如Brightness、Contrast等属性只适合相机。在使用Set方法设置视频捕获属性时,最好先使用Get方法获取当前视频捕获属性的值,以便确认是否支持该属性。

【代码位置:frmChapter9_1】Button3_Click

    Private Sub Button2_Click(sender As Object, e As EventArgs) Handles Button2.Click

        Dim vc As New VideoCapture("C:\learnEmgucv\movie1.mov")

        '获取当前视频宽度属性的值

        Dim width As Integer = vc.Get(CapProp.FrameWidth)

        '判断是否支持该属性

        If width > 0 Then

            '设置视频宽度为640像素

            vc.Set(CapProp.FrameWidth, 640)

        End If

        vc.Dispose()

End Sub

9.1.3 RetrieveReadQueryframe方法

这三种方法都是从视频中获得一幅图像。但是各有区别。

1、Retrieve不能自动读取下一帧,需要与Grab方法一起使用。而Grab方法只是获取下一帧的数据到一个用户无法访问的内存区,然后使用Retrieve方法根据通道号进行编码读取相应的图像。Read和Queryframe方法取出图像后可以自动转到下一帧图像

2、Retrieve可以返回多通道图像,比如使用立体摄像机,每一帧会返回两个图像;而Kinect会返回四个图像。Read和Queryframe方法不能读取多通道图像。

3、三种方法读取完视频(文件)最后一帧,返回的状态(详细请看示例代码):

  1. QueryFrame:IsNothing(m)=True(其中m为获得的Mat对象,下同),读取时,m所指向的数据指针DataPointer会变。
  2. Read:IsNothing(m)=False,m.IsEmpty=True,读取时,m所指向的数据指针DataPointer不会变。
  3. Retrieve:IsNothing(m)=False,m.IsEmpty= False,仍然会一直返回最后一帧的图像。因此需要判断是否达到最后一帧,即播放的当前帧是否是最后一帧。读取时,m所指向的数据指针DataPointer不会变。

由于上述三个方法以及Grab方法的使用都很简单,这里不再详述。

【代码位置:frmChapter9_1】Button4_Click

    '播放视频

    Private Sub Button4_Click(sender As Object, e As EventArgs) Handles Button4.Click

        Dim vc As New VideoCapture("C:\learnEmgucv\movie1.mp4")

        '判断是否已经打开

        If vc.IsOpened = False Then

            Exit Sub

        End If

        '获得帧率

        Dim fps As Double = vc.Get(CapProp.Fps)

        '获得总的帧数

        Dim framecount As Long = vc.Get(CapProp.FrameCount)

        Dim m As Mat = New Mat()

        Dim i As Long = 0

        While True

            '读取视频帧图像,三种方式:

            '==1、使用Grab+Retrieve

            If vc.Grab() = False Then

                Exit While

            End If

            If vc.Retrieve(m) = False Then

                Exit While

            End If

            '==2、使用Read

            'vc.Read(m)

            '==3、使用QueryFrame

            'm = vc.QueryFrame

            '如果m是空

            If IsNothing(m) Then

                Exit While

            End If

            '如果m未包含数据

            If m.IsEmpty Then

                Exit While

            End If

            '播放视频,两种方式

            '==1、在imagebox1里面播放

            '显示帧画面

            ImageBox1.Image = m

            '必须强制刷新,否则只会在结束时显示最后一帧画面

            ImageBox1.Refresh()

            '按照播放的帧率设置间隔时间

            Threading.Thread.Sleep(1000 \ fps)

            '==2、使用Imshow来播放

            '显示帧画面

            'CvInvoke.Imshow("video", m)

            '等待按键

            'CvInvoke.WaitKey(1000 / fps)

            '判断是否已经是结束的一帧

            If i >= framecount Then

                Exit While

            End If

            i += 1

            Label1.Text = i & "/" & framecount

            '必须强制刷新,否则只会看到最后的结果

            Label1.Update()

        End While

        '如果上述代码中使用了CvInvoke.Imshow来显示视频,那么这里应该销毁这个窗口

        'CvInvoke.DestroyWindow("video")

        vc.Dispose()

        Label1.Text = "播放结束"

End Sub

输出结果如下图所示:

图9-1 视频播放

在播放时设置视频播放的起始点,可以使用两种方式,都需要用到Set方法:

1、设置指定帧位置作为起始点:

vc.Set(CapProp.PosFrames, 帧位置)

2、设置指定时间位置作为起始点:

vc.Set(CapProp.PosMsec, 时间位置)

【代码位置:frmChapter9_1】Button5_Click

    '设置播放起始点

    Private Sub Button5_Click(sender As Object, e As EventArgs) Handles Button5.Click

        Dim vc As New VideoCapture("C:\learnEmgucv\movie1.mp4")

        vc.Set(CapProp.FrameWidth, 640)

        vc.Set(CapProp.FrameHeight, 480)

        Dim fps As Double = vc.Get(CapProp.Fps)

        Console.WriteLine(vc.Get(CapProp.FrameCount))

        If vc.IsOpened = False Then

            Exit Sub

        End If

        Dim framecount As Long = vc.Get(CapProp.FrameCount)

        Dim i As Long = 0

        '以下两种方式选其一:

        '==1、设置要播放的起始帧的位置

        Dim frameIndex As Integer = 50

        vc.Set(CapProp.PosFrames, frameIndex)

        '==2、设置要播放的事件点的位置,注意Set()的第二个参数是毫秒

        'Dim frameTime As Double = 3.0

        'vc.Set(CapProp.PosMsec, frameTime * 1000)

        Dim m As Mat = New Mat()

        While True

            '读取视频帧图像,使用Read

            vc.Read(m)

            If IsNothing(m) Then

                Exit While

            End If

            If m.IsEmpty Then

                Exit While

            End If

            CvInvoke.Imshow("video", m) ' 显示图像

            CvInvoke.WaitKey(1000 / fps) ' 等待按键

        End While

        CvInvoke.DestroyWindow("video")

        vc.Dispose()

        Label1.Text = "播放结束"

End Sub

事实上,当获得视频帧图像后,就可以对这个图像进行处理,再显示处理的结果。

【代码位置:frmChapter9_1】Button6_Click

    '简单示例处理图像后输出视频

    Private Sub Button6_Click(sender As Object, e As EventArgs) Handles Button6.Click

        Dim vc As New VideoCapture("C:\learnEmgucv\movie1.mp4")

        Dim fps As Double = vc.Get(CapProp.Fps)

        If vc.IsOpened = False Then

            Exit Sub

        End If

        Dim framecount As Long = vc.Get(CapProp.FrameCount)

        Dim i As Long = 0

        Dim m As Mat = New Mat()

        While True

            vc.Read(m)

            If IsNothing(m) Then

                Exit While

            End If

            If m.IsEmpty Then

                Exit While

            End If

            '这里使用canny简单示例处理图像

            Dim mcanny As New Mat

            CvInvoke.Canny(m, mcanny, 100, 250, 3)

            CvInvoke.Imshow("video", mcanny) ' 显示图像

            CvInvoke.WaitKey(1000 / fps) ' 等待按键

        End While

        CvInvoke.DestroyWindow("video")

        vc.Dispose()

        Label1.Text = "播放结束"

End Sub

输出结果如下图所示:

图9-2 处理视频帧并播放

9.1.4 StartStopPause方法

在EmguCV中,videocapture类提供了Start、Stop、Pause等方法,用于控制视频的播放和暂停。

Start方法:开始播放视频,实际是启动grab到一个指定线程。该方法并不会立即开始播放视频,而是会将视频设置为播放状态,此时将会触发ImageGrabbed事件。

Stop方法:停止播放视频,实际是停止grabbing线程。

Pause方法:暂停播放视频,实际是暂停正在运行的grab线程。该方法并不会立即暂停视频播放,而是会将视频设置为暂停状态。在Pause方法调用后,可以使用read或Grab方法读取视频帧进行处理,但是读取的视频帧是暂停状态下的帧图像,而不是实时的帧图像。

上述三个方法的使用,请参看9.1.5节【ImageGrabbed事件】的示例代码。

9.1.5 ImageGrabbed事件

ImageGrabbed事件会在每一帧图像数据被抓取时触发。使用该事件可以实现对视频流的实时处理。需要注意的是,1、ImageGrabbed事件实际已经是在进行Grab,所以,如果使用Retrieve方法获得图像,可以不使用Grab方法;2、由于ImageGrabbed事件是在后台线程中执行的,因此在事件处理方法中不能直接访问窗体或控件的属性。

【代码位置:frmChapter9_1】frmChapter9_Load、Button7_Click、vc1_ImageGrabbed、Button8_Click、Button9_Click

   Dim vc1 As VideoCapture

    '帧率

    Dim fps As Double = 0

    Private Sub frmChapter9_Load(sender As Object, e As EventArgs) Handles MyBase.Load

        '不检查跨线程

        Me.CheckForIllegalCrossThreadCalls = False

    End Sub

    '调用VideoCaptureImageGrabbed事件

    Private Sub Button7_Click(sender As Object, e As EventArgs) Handles Button7.Click

        vc1 = New VideoCapture("C:\learnEmgucv\movie1.mp4")

        If vc1.IsOpened = False Then

            MessageBox.Show("打开文件失败")

            Exit Sub

        End If

        '获得帧率

        fps = vc1.Get(CapProp.Fps)

        '添加ImageGrabbed事件

        AddHandler vc1.ImageGrabbed, AddressOf vc1_ImageGrabbed

        '启动

        vc1.Start()

    End Sub

    'ImageGrabbed事件,这里面进行播放

    Private Sub vc1_ImageGrabbed(sender As Object, e As EventArgs)

        Dim nextframe As New Mat

        '如果使用Retrieve,那么需要检查视频当前播放的位置

        vc1.Retrieve(nextframe)

        '判断是否到达视频结束帧

        If vc1.Get(CapProp.PosFrames) >= vc1.Get(CapProp.FrameCount) Then

            '停止

            vc1.Stop()

            '释放资源

            vc1.Dispose()

            '取消事件

            RemoveHandler vc1.ImageGrabbed, AddressOf vc1_ImageGrabbed

            '需要跨线程

            Label1.Text = "播放结束"

            Exit Sub

        End If

        '如果使用read,只需要检查该方法的返回值

        'If vc1.Read(nextframe) = False Then

        '    '停止

        '    vc1.Stop()

        '    '释放资源

        '    vc1.Dispose()

        '    '取消事件

        '    RemoveHandler vc1.ImageGrabbed, AddressOf vc1_ImageGrabbed

        '    '需要跨线程

        '    Label1.Text = "播放结束"

        '    Exit Sub

        'End If

        ImageBox1.Image = nextframe

        Threading.Thread.Sleep(1000 / fps)

    End Sub

    '暂停

    Private Sub Button8_Click(sender As Object, e As EventArgs) Handles Button8.Click

        vc1.Pause()

    End Sub

    '停止

    Private Sub Button9_Click(sender As Object, e As EventArgs) Handles Button9.Click

        vc2.Stop()

End Sub

以下代码模拟了电影中常用的扣除绿幕。

【代码位置:frmChapter9_1】Button10_Click、vc2_ImageGrabbed

   Dim vc2 As VideoCapture

    Private Sub Button10_Click(sender As Object, e As EventArgs) Handles Button10.Click

        vc2 = New VideoCapture("C:\learnEmgucv\前景.wmv")

        If vc2.IsOpened = False Then

            MessageBox.Show("打开文件失败")

            Exit Sub

        End If

        fps = vc2.Get(CapProp.Fps)

        AddHandler vc2.ImageGrabbed, AddressOf vc2_ImageGrabbed

        vc2.Start()

    End Sub

    '模拟绿幕

    Private Sub vc2_ImageGrabbed(sender As Object, e As EventArgs)

        Dim nextframe As New Mat

        '这里是对视频文件进行检查,所以使用read,同时检查是否播放完毕

        If vc2.Read(nextframe) = False Then

            vc2.Stop()

            vc2.Dispose()

            RemoveHandler vc2.ImageGrabbed, AddressOf vc2_ImageGrabbed

            '跨线程了

            Label1.Text = "播放结束"

            Exit Sub

        End If

        Dim mhsv As New Mat

        CvInvoke.CvtColor(nextframe, mhsv, ColorConversion.Rgb2Hsv)

        Dim lower As New ScalarArray(New MCvScalar(35, 43, 46))

        Dim upper As New ScalarArray(New MCvScalar(77, 255, 255))

        '提取图像中某个颜色范围内的像素

        '颜色值在范围内,则将其设置为白色(255),否则将其设置为黑色(0

        Dim mmask As New Mat

        CvInvoke.InRange(mhsv, lower, upper, mmask)

        '根据实际需要判断是否反转颜色

        Dim mreversalmask As New Mat

        mreversalmask = Not mmask

        '以下代码输出二值图作为mask的彩色图,也就是原图去除了绿色背景

        Dim m3channel As New Mat

        CvInvoke.CvtColor(mreversalmask, m3channel, ColorConversion.Gray2Bgr)

        Dim mout As New Mat

        CvInvoke.BitwiseAnd(nextframe, m3channel, mout)

        Threading.Thread.Sleep(1000 / fps)

        ImageBox1.Image = mout

End Sub

输出结果如下图所示:

 

图9-3 模拟去除绿幕

  • 10
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C# 使用 EmguCV 库实现双目测距的基本步骤涉及图像捕捉、立体匹配和深度图生成等几个关键环节。 ### 1. 图像捕捉 首先需要从两个摄像头捕获图像,这两个摄像头应该放置在固定的距离上,并尽可能平行对准同一场景区域。这通常涉及到驱动摄像头硬件并获取图像数据。 ```csharp using Emgu.CV; using Emgu.CV.CvEnum; // 初始化视频设备(摄像头) var capture = new VideoCapture(0); // 可能需要根据实际情况调整索引 while (true) { // 获取图像帧 var image = capture.QueryFrame(); if (image != null) { // 进行后续处理前显示图像 Console.WriteLine("Captured frame"); image.Show("Camera View"); // 按特定键退出循环 if (Console.KeyAvailable && Console.ReadKey().Key == ConsoleKey.Q) break; } } capture.Close(); ``` ### 2. 立体匹配算法 使用 EmguCV 实现双目视觉系统的关键是找到两幅图像之间的对应点,即左右摄像头视角下同一物体的不同投影。这个过程可以采用 SGBM( Stereo Block Matching Algorithm)、SSD( Sum of Squared Differences)等多种算法。 ```csharp using Emgu.CV.Structure; using Emgu.CV.Stereo; using Emgu.CV.Util; int windowSize = 5; // 调整窗口大小以优化精度与速度的平衡 StereoBM stereoMatcher = new StereoBM(windowSize, 16); Mat disparityMap = new Mat(); // 假设我们已经有了两个图像 leftImage 和 rightImage stereoMatcher.Compute(leftImage, rightImage, disparityMap); // 显示匹配结果,这里展示的是“disparity map”,表示每个像素到最近平面的距离 disparityMap.Show("Disparity Map"); ``` ### 3. 深度图生成 从 disparity map 中提取深度信息,进而计算出各个点相对于相机系统的距离。 ```csharp public double DepthInMeter(int row, int col, double baseline) { double disp = disparityMap.Get<float>(row, col) / 16f; return baseline * focalLength / (disp + 1e-9); } double baseline = ...; // 需要测量的两个摄像头间的物理基线长度 double focalLength = ...; // 相机内参中的焦距 for (int i = 0; i < disparityMap.Height; i++) { for (int j = 0; j < disparityMap.Width; j++) { double depth = DepthInMeter(i, j, baseline); Console.WriteLine($"Depth at ({j}, {i}): {depth} meters"); } } ``` ### 4. 结果应用 最后,基于上述步骤得到的深度信息,可以进行物体定位、尺寸测量等进一步的应用分析。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值