(一)Qt下实现多个海康工业相机内触发采集回调取流显示

news2024/7/6 17:51:42

系列文章目录

提示:这里是该系列文章的所有文章的目录
第一章:(一)Qt下实现多个海康工业相机内触发采集回调取流显示
第二章:(二)Qt下多线程实现多个海康工业相机内触发采集回调取流显示


文章目录

  • 系列文章目录
  • 前言
  • 一、环境配置
  • 二、图像显示流程
  • 三、示例完整代码
  • 四、下载链接
  • 总结


前言

在我之前所记录的关于海康工业相机的系列文章中 ,讲述的是使用外触发采集模式中的软触发,并采用超时机制获取一帧图片主动取流的,但在后期的测试过程中发现会出现失帧的的情况,所以决定采用内触发采集模式,并使用回调取流的方式来获取图像。本文主要讲述了在Qt下使用回调取流的方式来实现两个相机的显示,并结合相应的示例进行讲解,以便大家学习,如有错误之处,欢迎大家批评指正。

项目效果
请添加图片描述


提示:以下是本篇文章正文内容,下面案例可供参考

一、环境配置

同样的,要在Qt下实现海康工业相机的采集,需要将相关SDK集成到项目中去,这里按照前文使用pri的方式将海康SDK添加到pro项目中去,具体的可查看该篇文章(一)Qt+OpenCV调用海康工业相机SDK示例开发,我的示例项目是MSVC 64位,所以在pro中也是添加了字符的设置:

#设置字符
contains( CONFIG,"msvc" ):QMAKE_CXXFLAGS += /source-charset:utf-8 /execution-charset:utf-8
contains( CONFIG,"msvc" ):QMAKE_CFLAGS +=/source-charset:utf-8 /execution-charset:utf-8

#海康SDK
include (./HikSdk/HikSdk.pri)

示例项目结构如下:
请添加图片描述

二、图像显示流程

海康工业相机使用回调取流的整个流程如下:
1.枚举所连接的相机信息列表,我这里进行了GIGE和USB类型的相机的枚举
2.可以获取相机序列号来指定相机设备并打开相机
3.使用SetEnumValue(“TriggerMode”,0)关闭外触发模式
4.在开启采集前进行回调函数的注册
5.开启采集,并在回调函数中将图像数据转换为QImage类型并显示到界面

//初始化相机对象,完成开启采集前的流程
void MainWindow::initWidget()
{
    //相机对象
    for(int i=0;i<2;i++)
    {
        m_myCamera[i] = new CMvCamera;
    }

    //枚举子网内所有设备
    memset(&m_stDevList,0,sizeof(MV_CC_DEVICE_INFO_LIST));
    int nRet = CMvCamera::EnumDevices(MV_GIGE_DEVICE | MV_USB_DEVICE,&m_stDevList);
    if(MV_OK != nRet)
    {
        LOGDEBUG<<"枚举相机设备失败!";
        return;
    }

    int deviceNum = m_stDevList.nDeviceNum;
    LOGDEBUG<<"deviceNum:"<<deviceNum;
    for(int i=0;i<deviceNum;i++)
    {
        MV_CC_DEVICE_INFO *pDeviceInfo = m_stDevList.pDeviceInfo[i];
        QString strSerialNumber = "";
        if(pDeviceInfo->nTLayerType == MV_GIGE_DEVICE)
        {
            strSerialNumber = (char*)pDeviceInfo->SpecialInfo.stGigEInfo.chSerialNumber;
        }
        else if(pDeviceInfo->nTLayerType == MV_USB_DEVICE)
        {
            strSerialNumber = (char*)pDeviceInfo->SpecialInfo.stUsb3VInfo.chSerialNumber;
        }
        else
        {
            LOGDEBUG<<"警告,未知设备枚举!";
            return;
        }
        LOGDEBUG<<"i:"<<i<<"   strSerialNumber:"<<strSerialNumber;

        //根据相机序列号指定相机对象
        //if(strSerialNumber == "DA0333897")
        //{
        //    m_deviceInfo[0] = pDeviceInfo;
        //}
        //else if(strSerialNumber == "DA0424312")
        //{
        //    m_deviceInfo[1] = pDeviceInfo;
        //}

        //不指定
        m_deviceInfo[i] = pDeviceInfo;

        //打开相机
        int nRet = m_myCamera[i]->Open(m_deviceInfo[i]);
        if(MV_OK != nRet)
        {
            LOGDEBUG<<"i:"<<i<<"打开相机失败!";
            return;
        }

        //关闭触发模式
        nRet = m_myCamera[i]->SetEnumValue("TriggerMode",0);
        if(MV_OK != nRet)
        {
            LOGDEBUG<<"i:"<<i<<"关闭触发模式失败!";
            return;
        }

        //注册回调函数
        //nRet = m_myCamera[i]->RegisterImageCallBack(ImageCallback,this);    //单色相机
        //nRet = m_myCamera[i]->RegisterImageCallBackRGB(ImageCallback_1,this);   //彩色相机
        //if(MV_OK != nRet)
        //{
        //    LOGDEBUG<<"i:"<<i<<"注册回调函数失败!";
        //    return;
        //}
    }

    //注册回调函数
    nRet = m_myCamera[0]->RegisterImageCallBackRGB(ImageCallback_1,this);
    if(MV_OK != nRet)
    {
        LOGDEBUG<<"相机1注册回调函数失败!";
        return;
    }

    nRet = m_myCamera[1]->RegisterImageCallBackRGB(ImageCallback_2,this);
    if(MV_OK != nRet)
    {
        LOGDEBUG<<"相机2注册回调函数失败!";
        return;
    }
}

//回调函数1
void __stdcall MainWindow::ImageCallback_1(unsigned char * pData,MV_FRAME_OUT_INFO_EX* pFrameInfo,void* pUser)
{
    LOGDEBUG<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss:zzz ")<<"回调函数1执行了";
    MainWindow* pThis = (MainWindow*)pUser;
    QImage showImage = QImage(pData, pFrameInfo->nWidth,pFrameInfo->nHeight,QImage::Format_RGB888);
    pThis->showImage(showImage,1);
}

三、示例完整代码

1.HikSdk.pri整个文件夹中有相应的头文件源文件及dll,可直接通过后文的下载链接进行下载,这里对修改的一处进行展示,其它的在这里就不做展示了(也可以在之前的海康系列文章中查看下)

//cmvcamera.h
// ch:注册彩色图像数据回调
int RegisterImageCallBackRGB(void(__stdcall* cbOutput)(unsigned char * pData, MV_FRAME_OUT_INFO_EX* pFrameInfo, void* pUser), void* pUser);

//cmvcamera.cpp
// ch:注册彩色图像数据回调
int CMvCamera::RegisterImageCallBackRGB(void(__stdcall* cbOutput)(unsigned char * pData, MV_FRAME_OUT_INFO_EX* pFrameInfo, void* pUser), void* pUser)
{
    return MV_CC_RegisterImageCallBackForRGB(m_hDevHandle, cbOutput, pUser);
}

2.mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QDateTime>
#include "HikSdk/cmvcamera.h"

QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

    void initWidget();

    void showImage(QImage showImage,int index);

private slots:
    void on_pb_start_clicked();
    void on_pb_stop_clicked();

private:
    static void __stdcall ImageCallback_1(unsigned char * pData,MV_FRAME_OUT_INFO_EX* pFrameInfo,void* pUser);
    static void __stdcall ImageCallback_2(unsigned char * pData,MV_FRAME_OUT_INFO_EX* pFrameInfo,void* pUser);

private:
    Ui::MainWindow *ui;

    CMvCamera *m_myCamera[2];             //相机对象
    MV_CC_DEVICE_INFO *m_deviceInfo[2];   //设备信息
    MV_CC_DEVICE_INFO_LIST m_stDevList;   //设备信息列表
};
#endif // MAINWINDOW_H

3.mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    this->initWidget();
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::initWidget()
{
    //相机对象
    for(int i=0;i<2;i++)
    {
        m_myCamera[i] = new CMvCamera;
    }

    //枚举子网内所有设备
    memset(&m_stDevList,0,sizeof(MV_CC_DEVICE_INFO_LIST));
    int nRet = CMvCamera::EnumDevices(MV_GIGE_DEVICE | MV_USB_DEVICE,&m_stDevList);
    if(MV_OK != nRet)
    {
        LOGDEBUG<<"枚举相机设备失败!";
        return;
    }

    int deviceNum = m_stDevList.nDeviceNum;
    LOGDEBUG<<"deviceNum:"<<deviceNum;
    for(int i=0;i<deviceNum;i++)
    {
        MV_CC_DEVICE_INFO *pDeviceInfo = m_stDevList.pDeviceInfo[i];
        QString strSerialNumber = "";
        if(pDeviceInfo->nTLayerType == MV_GIGE_DEVICE)
        {
            strSerialNumber = (char*)pDeviceInfo->SpecialInfo.stGigEInfo.chSerialNumber;
        }
        else if(pDeviceInfo->nTLayerType == MV_USB_DEVICE)
        {
            strSerialNumber = (char*)pDeviceInfo->SpecialInfo.stUsb3VInfo.chSerialNumber;
        }
        else
        {
            LOGDEBUG<<"警告,未知设备枚举!";
            return;
        }
        LOGDEBUG<<"i:"<<i<<"   strSerialNumber:"<<strSerialNumber;

        //根据相机序列号指定相机对象
        //if(strSerialNumber == "DA0333897")
        //{
        //    m_deviceInfo[0] = pDeviceInfo;
        //}
        //else if(strSerialNumber == "DA0424312")
        //{
        //    m_deviceInfo[1] = pDeviceInfo;
        //}

        //不指定
        m_deviceInfo[i] = pDeviceInfo;

        //打开相机
        int nRet = m_myCamera[i]->Open(m_deviceInfo[i]);
        if(MV_OK != nRet)
        {
            LOGDEBUG<<"i:"<<i<<"打开相机失败!";
            return;
        }

        //关闭触发模式
        nRet = m_myCamera[i]->SetEnumValue("TriggerMode",0);
        if(MV_OK != nRet)
        {
            LOGDEBUG<<"i:"<<i<<"关闭触发模式失败!";
            return;
        }

        //注册回调函数
        //nRet = m_myCamera[i]->RegisterImageCallBack(ImageCallback,this);    //单色相机
        //nRet = m_myCamera[i]->RegisterImageCallBackRGB(ImageCallback_1,this);   //彩色相机
        //if(MV_OK != nRet)
        //{
        //    LOGDEBUG<<"i:"<<i<<"注册回调函数失败!";
        //    return;
        //}
    }

    //注册回调函数
    nRet = m_myCamera[0]->RegisterImageCallBackRGB(ImageCallback_1,this);
    if(MV_OK != nRet)
    {
        LOGDEBUG<<"相机1注册回调函数失败!";
        return;
    }

    nRet = m_myCamera[1]->RegisterImageCallBackRGB(ImageCallback_2,this);
    if(MV_OK != nRet)
    {
        LOGDEBUG<<"相机2注册回调函数失败!";
        return;
    }

}

void MainWindow::showImage(QImage showImage,int index)
{
    QPixmap showPixmap = QPixmap::fromImage(showImage).scaled(QSize(250,200),Qt::IgnoreAspectRatio,Qt::SmoothTransformation);
    if(index == 1)
    {
        ui->lb_image_1->setPixmap(showPixmap);
    }
    else
    {
        ui->lb_image_2->setPixmap(showPixmap);
    }
}

void MainWindow::on_pb_start_clicked()
{
    //开始取图
    for(int i=0;i<2;i++)
    {
        int nRet = m_myCamera[i]->StartGrabbing();
        if (MV_OK != nRet)
        {
            LOGDEBUG<<"i:"<<i<<"开始取图失败!";
            return;
        }
    }
}

void MainWindow::on_pb_stop_clicked()
{
    //停止取图
    for(int i=0;i<2;i++)
    {
        int nRet = m_myCamera[i]->StopGrabbing();
        if (MV_OK != nRet)
        {
            LOGDEBUG<<"i:"<<i<<"停止取图失败!";
            return;
        }
    }
}

//回调函数1
void __stdcall MainWindow::ImageCallback_1(unsigned char * pData,MV_FRAME_OUT_INFO_EX* pFrameInfo,void* pUser)
{
    LOGDEBUG<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss:zzz ")<<"回调函数1执行了";
    MainWindow* pThis = (MainWindow*)pUser;
    QImage showImage = QImage(pData, pFrameInfo->nWidth,pFrameInfo->nHeight,QImage::Format_RGB888);
    pThis->showImage(showImage,1);
}

//回调函数2
void __stdcall MainWindow::ImageCallback_2(unsigned char * pData,MV_FRAME_OUT_INFO_EX* pFrameInfo,void* pUser)
{
    LOGDEBUG<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss:zzz ")<<"回调函数2执行了";
    MainWindow* pThis = (MainWindow*)pUser;
    QImage showImage = QImage(pData, pFrameInfo->nWidth,pFrameInfo->nHeight,QImage::Format_RGB888);
    pThis->showImage(showImage,2);
}

4.mainwindow.ui
在这里插入图片描述

四、下载链接

我的示例百度网盘链接:https://pan.baidu.com/s/1J35z54pcP5k0Ss8jlcIY1g
提取码:xxcj


总结

可以看到Qt下连接海康工业相机使用回调取流的整个流程就是这样的,注意一点就是要在打开相机后才能修改相机的各个参数,使用回调取流的方式,会根据你相机的帧率来获取图像的。在这个示例中,使用了两个相机,结果需要两个回调函数来获取对应的图像,如果相机个数更多个呢?那这样是不是就不太方便了,大家可以思考下如何改进下这个问题呢?


hello:
共同学习,共同进步,如果还有相关问题,可在评论区留言进行讨论。

参考博客:海康工业相机采集图像的流程和采集模式分类

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/757819.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何设计一个注册中心?(1)概念

1. 为什么需要注册中心 一个集群中有众多服务&#xff0c;每个服务有N个实例&#xff0c;因此需要一个第三方节点来存放每个服务的信息&#xff0c;这样服务需要其它的服务信息&#xff0c;直接到第三方节点获取就行了。这个第三方的节点&#xff0c;就是注册中心。 2. 注册中…

2023.7.14 智慧芽前端面试总结

整体面试体验感蛮不错的&#xff0c;面试官很愿意与我交流&#xff0c;面试的结尾也给出了一定的学习建议。 由于这家公司主要的业务是做搜索引擎方面&#xff0c;估摸着是个自研。面试注重点主要是偏业务处理和针对工具的使用&#xff0c;还有无障碍阅读英文文章&#xff0c;…

C++进阶—哈希/unordered系列关联式容器/底层结构(一篇文章学习哈希)

目录 0. 前言map/set和unordered_map/unordered_set 1. unordered系列关联式容器 1.1 unordered_map 1.1.2 unordered_map的接口说明 1. unordered_map的构造 2. unordered_map的容量 3. unordered_map的迭代器 4. unordered_map的元素访问 5. unordered_map的查询 6…

Spring原码学习第一篇:Spring源码概述

1、Spring获取对象的过程 2、Spring源码概述图&#xff1a; 2、一些重要的接口 BeanDefinition中实现的方法&#xff0c;把xml中定义的对象封装为一个对象&#xff0c;方便后面处理 4、BeandefinitionReader BeandefinitionReader作为一个抽象层来处理配置文件&#xff0c;定…

Autosar通信实战系列01-CanSM模块功能及配置开发详解

本文框架 前言1. CanSM General配置2. 通道详细配置解析2.1 添加一路CanSMController2.2 CanSMController配置2.3 关联DemEvent配置前言 在本系列笔者将结合工作中对通信实战部分的应用经验进一步介绍常用,包括但不限于通信各模块的开发教程,代码逻辑分析,调测试方法及典型…

【Maven二】——maven仓库

系列文章目录 Maven之POM介绍 maven命令上传jar包到nexus maven仓库 系列文章目录前言一、什么是maven仓库&#xff1f;二、仓库的分类建立私服的优势 三、远程仓库的配置四、远程仓库的认证部署至远程仓库五、快照版本&why六、从仓库解析依赖的机制七、镜像总结 前言 由…

习题—实用类

目录 1.xxx 1.包装类及其构造方法 import java.util.List; import java.util.ArrayList; import java.util.Scanner;//包装类及其构造方法 public class Test {public static void main(String[] args) {// 装箱&#xff1a;基本类型转换为包装类的对象 // 拆箱&#xff1a…

go mod verdor简明介绍

Go 语言在 go 1.6 版本以后编译 go 代码会优先从 vendor 目录先寻找依赖包&#xff0c;它具有以下优点&#xff1a; 复制依赖&#xff1a;go mod vendor 会把程序所依赖的所有包复制到项目目录下的vendor 文件夹中&#xff0c;所以即使这些依赖包在外部源&#xff08;如 GitHu…

「C/C++」C/C++宏定义#define

✨博客主页&#xff1a;何曾参静谧的博客 &#x1f4cc;文章专栏&#xff1a;「C/C」C/C程序设计 目录 术语说明宏定义 #define定义常量定义函数定义代码块常用标识符用宏包含头文件 术语说明 定义宏是一种预处理器指令&#xff0c;它可以将一些代码片段或者常量直接替换为另一…

刘二大人Pytorch课程笔记

Lecture01. Overview 没啥好记的&#xff0c;理解就好 人工智能和机器学习等的关系&#xff1a; 正向传播 正向传播本质上是按照输入层到输出层的顺序&#xff0c;求解并保存网络中的中间变量本身。 反向传播 反向传播本质上是按照输出层到输入层的顺序&#xff0c;求解并…

LangChain 联合创始人下场揭秘:如何用 LangChain 和向量数据库搞定语义搜索?

近期&#xff0c;关于 ChatGPT 的访问量有所下降的消息引发激烈讨论&#xff0c;不过这并不意味着开发者对于 AIGC 的热情有所减弱&#xff0c;例如素有【2023 最潮大语言模型 Web 开发框架】之称的大网红 LangChain 的热度就只增不减。 原因在于 LangChain 作为大模型能力“B2…

1快速入门MyBatis

开发前的准备 准备数据库表&#xff1a;汽⻋表t_car 确定表中的字段以及字段的数据类型 guide_price是decimal类型&#xff0c;专⻔为财务数据准备的类型produce_time可以用char类型 , 格式’2022-10-11’ 使用navicat for mysql⼯具向t_car表中插⼊两条数据 配置IDEA中ma…

【C++修炼之路】vector 模拟实现

&#x1f451;作者主页&#xff1a;安 度 因 &#x1f3e0;学习社区&#xff1a;StackFrame &#x1f4d6;专栏链接&#xff1a;C修炼之路 文章目录 一、读源码二、成员变量三、默认成员函数1、构造2、析构3、拷贝构造4、赋值重载 四、访问1、[ ] 重载2、迭代器 五、容量1、cap…

Profibus DP主站转Modbus TCP网关profibus从站地址范围

远创智控YC-DPM-TCP网关。这款产品在Profibus总线侧实现了主站功能&#xff0c;在以太网侧实现了ModbusTcp服务器功能&#xff0c;为我们的工业自动化网络带来了全新的可能。 远创智控YC-DPM-TCP网关是如何实现这些功能的呢&#xff1f;首先&#xff0c;让我们来看看它的Profib…

Oracle解析JSON字符串

Oracle解析JSON字符串 假设某个字段存储的JSON字符串&#xff0c;我们不想查出来后通过一些常见的编程语言处理&#xff08;JSON.parse()或者是JSONObject.parseObject()等&#xff09;&#xff0c;想直接在数据库上处理&#xff0c;又该如何书写呢&#xff1f; 其实在ORACLE中…

算法06-搜索算法-广度优先搜索

文章目录 参考&#xff1a;总结大纲要求搜索算法-广度优先搜索迷宫问题问题迷宫的存储迷宫的移动搜索方式代码实现 图的广度优先遍历题目描述用邻接矩阵表示图 搜索算法-广度优先搜索 参考&#xff1a; 【算法设计】用C类和队列实现图搜索的广度优先遍历算法 C/C 之 广度优先…

梯度下降(Gradient Descent)

基本思想 梯度下降是一个用来求函数最小值的算法&#xff0c;本次&#xff0c;我们将使用梯度下降算法来求出代价函数的最小值。 梯度下降背后的思想是&#xff1a;开始时我们随机选择一个参数的组合&#xff0c;计算代价函数&#xff0c;然后我们寻找下一个能让代价函数值下降…

Linux:squid透明代理

在传统代理上进行修改并添加网卡 这次不使用手动代理&#xff0c;而是把网关搞成代理 在下面这个链接里的文章实验下进行修改 Linux&#xff1a;squid传统代理_鲍海超-GNUBHCkalitarro的博客-CSDN博客 完成以后不用再win10上去配置&#xff0c;代理的那一步&#xff0c;然后…

Python(十二)常见的数据类型

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…

TabLayout+ViewPager实现滚动页面

目录 一、TabLayout介绍 二、TabLayout的常用属性和方法 常用属性&#xff1a; 常用方法&#xff1a; 三、适配器介绍 &#xff08;一&#xff09;、PagerAdapter介绍&#xff1a; &#xff08;二&#xff09;、FragmentPagerAdapter介绍&#xff1a; &#xff08;三&am…