上一篇实现了在MFC的窗体内显示图片,本篇介绍如何在MFC窗体内实时显示摄像头的影像。

要实现的功能是点击一个“开始”按钮,可以显示影像,再点击“停止”按钮,可以停止显示。

因为实时显示影像需要在一个循环里执行,为了在显示影像的同时还可以干别的(比如,点击“停止”按钮),这里需要用到多线程,即显示影像的代码放到子线程中,与主线程并发执行。

重点已经说清楚了,下面是开发步骤:

1、先把Halcon中实时显示的程序搞定

2、Halcon代码导出为C++代码

3、建立MFC工程

4、在MFC中添加Halcon代码

下面说细节:

1、打开Halcon,点击助手,选择打开新的Acquisition

从资源选项卡可以看到检测到的接口为DirectShow,这是微软开发的视频设备驱动。

从连接选项卡能看到检测到相机,是笔记本自带的摄像头。点击上方的摄像机图标,Halcon的图形窗口就开始实时显示摄像头的画面了,很方便。

下面点击“代码生成”选项卡,点击“插入代码”按钮,就把实时显示的代码插入到代码窗口中了。

注意这里的采集模式是异步采集,在循环中采集图像的意思就是实时显示。

生成的Halcon代码如下:

* Image Acquisition 01: Code generated by Image Acquisition 01
open_framegrabber ('DirectShow', 1, 1, 0, 0, 0, 0, 'default', 8, 'rgb', -1, 'false', 'default', '[0] Integrated Camera', 0, -1, AcqHandle)
grab_image_start (AcqHandle, -1)
while (true)
grab_image_async (Image, AcqHandle, -1)
* Image Acquisition 01: Do something
endwhile
close_framegrabber (AcqHandle)
 

2、导出Halcon代码

导出的C++代码中Action函数如下:

void action()
{
// Local iconic variables
HObject ho_Image;
// Local control variables
HTuple hv_AcqHandle;
//Image Acquisition 01: Code generated by Image Acquisition 01
OpenFramegrabber("DirectShow", 1, 1, 0, 0, 0, 0, "default", 8, "rgb", -1, "false",
"default", "[0] Integrated Camera", 0, -1, &hv_AcqHandle);
GrabImageStart(hv_AcqHandle, -1);
while (0 != 1)
{
GrabImageAsync(&ho_Image, hv_AcqHandle, -1);
//Image Acquisition 01: Do something
}
CloseFramegrabber(hv_AcqHandle);
}

这就是我们需要添加到MFC的代码,需要注意到while循环中只是获取了图像,并没有显示图像,所以我们还要添加显示图像的代码。

3、建立MFC工程

与上一篇类似,新建基于对话框的MFC项目,添加Picture Control 和两个按钮。

4、添加C++代码

首先打开对话框类的头文件HalconCameraDlg.h,需要做下面四件事:

1、在文件开头添加Halcon头文件以及命名空间

#include "halconcpp.h"
using namespace HalconCpp;

2、在类外添加线程函数的声明

//线程函数的声明应在类CMultiThread1Dlg的外部

void ThreadFunc(LPVOID lpParam);

3、在类内添加Halcon变量为对话框类的Public成员

HObject  ho_Image;

HTuple  hv_AcqHandle;

HTuple m_HWindowID;

HTuple m_FGHandle,m_ImageWidth,m_ImageHeight;

4、添加线程函数的变量为对话框类的Protected成员

HANDLE hThread;

DWORD ThreadID;

然后在HalconCameraDlg.cpp中添加代码:

1、首先添加Halcon头文件和命名空间,并定义全局变量

volatile BOOL m_bRun;

volatile BOOL m_bShowFlag;

Volatile关键词告诉编译器不对此变量进行优化,使该值可被多个线程修改,对于多线程意义重大。

2、为开始按钮添加单机响应函数

	CWnd * pWnd = AfxGetApp()->GetMainWnd();
if(m_bShowFlag){
m_bRun=TRUE;
}else{
hThread = CreateThread(NULL,
0,
(LPTHREAD_START_ROUTINE) ThreadFunc, //
this, //传入主窗口指针
0,
&ThreadID );
}
为了能随时停止和开始实时监控,我设置了m_bShowFlag这个变量,第一次点击“开始”按钮时,m_bShowFlag为FALSE,执行CreateThread函数启动子线程,在子线程中m_bShowFlag被置为TRUE,所以下次点击“开始”按钮时不会再次开启子线程,而只是修改线程中的标志位来启动实时监控。

3、子线程函数的实现代码

void ThreadFunc(LPVOID pParam)
{
CHalconCameraDlg * pMainWindow;
pMainWindow=(CHalconCameraDlg * ) pParam; //强制转化为主窗口指针
HTuple HWindowID;
CRect Rect;
CWnd * pWnd = pMainWindow->GetDlgItem( IDC_STATIC);
HWindowID = (Hlong)pWnd->m_hWnd;
pWnd->GetWindowRect(&Rect);
OpenWindow(0,0,Rect.Width(),Rect.Height(),HWindowID,"visible","",&(pMainWindow->m_HWindowID) );
//显示相机捕捉的图像
OpenFramegrabber("DirectShow", 1, 1, 0, 0, 0, 0, "default", 8, "rgb", -1, "false",
"default", "[0] Integrated Camera", 0, -1, &(pMainWindow->hv_AcqHandle) );
GrabImageStart(pMainWindow->hv_AcqHandle, -1);
ClearWindow(pMainWindow->m_HWindowID);
GrabImage(&(pMainWindow->ho_Image), pMainWindow->hv_AcqHandle);
GetImagePointer1((pMainWindow->ho_Image),NULL,NULL,&(pMainWindow->m_ImageWidth),&(pMainWindow->m_ImageHeight) );
SetPart(pMainWindow->m_HWindowID,0,0,pMainWindow->m_ImageHeight-1,pMainWindow->m_ImageWidth-1);
m_bShowFlag=TRUE;//设置运行状态
m_bRun=TRUE;
while (m_bShowFlag){
if(m_bRun){
GrabImageAsync(&(pMainWindow->ho_Image), pMainWindow->hv_AcqHandle, -1);
DispObj(pMainWindow->ho_Image, pMainWindow->m_HWindowID);
Sleep(50);
}
}
ClearWindow(pMainWindow->m_HWindowID);
CloseFramegrabber(pMainWindow->hv_AcqHandle);
CloseWindow(pMainWindow->m_HWindowID);
ExitThread(0);
} while循环之前的代码与上一篇类似,循环中当m_bRun为TRUE时执行获取与显示图像的语句,因此当全局变量m_bRun被置为FALSE时显示会停止,实现了前述的功能(注意,此时线程并不退出)。

4、停止按钮的响应函数

只需要一句话就够了。因为m_bRun被声明为volatile变量,在子线程外部可以更改它,修改为FALSE之后子线程中实时显示的语句就无法执行,表现出来就是图像静止,不再更新。

m_bRun=FALSE;

遇到的问题:

在这个程序中,子线程一直没有退出,即m_bShowFlag没有被置为FLASE。

之前我试过在停止按钮里把m_bShowFlag置为FALSE,即让线程退出,然后再次点击开始按钮时重新启动线程,但是在关闭窗口时会出现下面的错误。

触发了一个断点。其原因可能是堆被损坏。原因也可能是用户在HalconCamera.exe具有焦点时按下了F12。

这个错误可能是退出线程时没有把空间释放干净所致,在多次的开启与关闭子线程(即多次点击开始和停止按钮)后,就会出现问题。

所以只能改为现在的线程不退出方案,让子线程一直执行,通过修改其中的标志位来启动和停止显示。

MFC&Halcon之实时视频监控的更多相关文章

  1. MFC&Halcon之图片显示

    上一篇介绍了MFC调用Halcon功能的简单例子,在这个例子中,点击按钮会弹出一个窗口显示图片,但是这样显示并不美观,而且...于是我决定把显示窗口做在MFC的窗体内部. 具体的方案是:使用一个Pic ...

  2. 【嵌入式开发】树莓派h264实时视频监控

    FishXX之前用VLC串流输出视频,在电脑上需要VLC播放器.试了一下,感觉还是有点延时. 今天发现一个更加流畅,也不需要VLC播放器的树莓派远程视频监控方法,直接在浏览器中打开即可. 地址:htt ...

  3. linux下motion摄像头监控编译与配置

    利用linxu下的开源的motion搭建嵌入式视频动态监控系统 所谓移动图像监测,简单来说就是利用摄像头定点监测某个区域,当有移动物体经过时,摄像头便自动抓拍(要监测多大物体.按拍照速率都是可调的), ...

  4. c# 远程监控(1) 大纲

    此篇文章主要讲了如何使用C# Winform程序模拟一个远程监控系统,并使用RTP实时传输协议传输数据. 应用场景:远程监控.局域网视频会议.客户端流式缓冲播放 这方面的资料还是有一些,但是都需要整合 ...

  5. Raspberry Pi + 3个USB摄像头 + Motion(简易监控设备配置记录1——介绍以及安装) 分类: Raspberry Pi 服务器搭建 2015-04-12 19:21 226人阅读 评论(0) 收藏

    参考: Debian官网链接 Motion官网链接 首先,参见Debian官网链接对Motion的介绍,网页中包含了所有相关依赖包,请首先确保这些依赖包的安装. Motion介绍 摘出对Motion的 ...

  6. 13行代码实现:Python实时视频采集(附源码)

    一.前言 本文是<人脸识别完整项目实战>系列博文第3部分:程序设计篇(Python版),第1节<Python实时视频采集程序设计>,本章内容系统介绍:基于Python+open ...

  7. 基于Spring4+SpringMVC4+Mybatis3+Hibernate4+Junit4框架构建高性能企业级的部标1077视频监控平台

    开发企业级的部标GPS监控平台,投入的开发力量很大,开发周期也很长,选择主流的开发语言以及成熟的开源技术框架来构建基础平台,是最恰当不过的事情,在设计之初就避免掉了技术选型的风险,避免以后在开发过程中 ...

  8. 基于C#和Asp.NET MVC开发GPS部标视频监控平台

    基于C#和Asp.NET MVC开发GPS部标监控平台 目前整理了基于.NET技术的部标平台开发文章,可以参考: 1.部标Jt808协议模拟终端的设计和开发 2.C#版的808GPS服务器开发-> ...

  9. GIS和视频监控的集成

    本文讨论了使用增强现实(AR)技术的三维(3D)地理信息系统(GIS)和视频监视系统的集成.进行这种集成的动机是要克服常规视频监视系统面临的问题.关于哪个摄像机当前监视此类系统中哪个区域的明确信息:因 ...

随机推荐

  1. UITextView 输入字数限制

    本文介绍了UITextView对中英文还有iOS自带表情输入的字数限制,由于中文输入会有联想导致字数限制不准确所以苦恼好久,所以参考一些大神的博客终于搞定,欢迎大家参考和指正. 对于限制UITextV ...

  2. Java 8 的 Nashorn 脚本引擎教程

    本文为了解所有关于 Nashorn JavaScript 引擎易于理解的代码例子. Nashorn JavaScript 引擎是Java SE 8的一部分,它与其它像Google V8 (它是Goog ...

  3. .NET Core的日志[4]:将日志写入EventLog

    面向Windows的编程人员应该不会对Event Log感到陌生,以至于很多人提到日志,首先想到的就是EventLog.EventLog不仅仅记录了Windows系统自身针对各种事件的日志,我们的应用 ...

  4. .NET Core的日志[1]:采用统一的模式记录日志

    记录各种级别的日志是所有应用不可或缺的功能.关于日志记录的实现,我们有太多第三方框架可供选择,比如Log4Net.NLog.Loggr和Serilog 等,当然我们还可以选择微软原生的诊断框架(相关A ...

  5. redis集成到Springmvc中及使用实例

    redis是现在主流的缓存工具了,因为使用简单.高效且对服务器要求较小,用于大数据量下的缓存 spring也提供了对redis的支持: org.springframework.data.redis.c ...

  6. 多线程爬坑之路-Thread和Runable源码解析

    多线程:(百度百科借一波定义) 多线程(英语:multithreading),是指从软件或者硬件上实现多个线程并发执行的技术.具有多线程能力的计算机因有硬件支持而能够在同一时间执行多于一个线程,进而提 ...

  7. JS实现页面进入、返回定位到具体位置

    最为一个刚入职不久的小白...慢慢磨练吧... JS实现页面返回定位到具体位置 其实浏览器也自带了返回的功能,也就是说,自带了返回定位的功能.正常的跳转,返回确实可以定位,但是有些特殊场景就不适用了. ...

  8. VSCode调试go语言出现:exec: "gcc": executable file not found in %PATH%

    1.问题描述 由于安装VS15 Preview 5,搞的系统由重新安装一次:在用vscdoe编译go语言时,出现以下问题: # odbcexec: "gcc": executabl ...

  9. 代码的坏味道(16)——纯稚的数据类(Data Class)

    坏味道--纯稚的数据类(Data Class) 特征 纯稚的数据类(Data Class) 指的是只包含字段和访问它们的getter和setter函数的类.这些仅仅是供其他类使用的数据容器.这些类不包 ...

  10. Docker与CI持续集成/CD

    背景        Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化.容器是完全使用沙箱机制 ...