Baumer工业相机堡盟相机如何使用JPEG图像压缩功能(LXT.JP系列相机图像压缩功能的使用和优点以及行业应用)(C++)

简介: Baumer工业相机堡盟相机如何使用JPEG图像压缩功能(LXT.JP系列相机图像压缩功能的使用和优点以及行业应用)(C++)

项目场景

Baumer工业相机堡盟相机是一种高性能、高质量的工业相机,可用于各种应用场景,如物体检测、计数和识别、运动分析和图像处理。  


Baumer的万兆网相机拥有出色的图像处理性能,可以实时传输高分辨率图像。此外,该相机还具有快速数据传输、低功耗、易于集成以及高度可扩展性等特点。  


Baumer万兆网相机中LXT.JP图像压缩系列相机是一种预处理相机,在相机内部对图像进行JPEG算法压缩然后再传输到处理器中。


技术背景

工业相机的JPEG图像压缩功能有助于减少图像文件的大小,同时保持图像质量。这种压缩功能是基于JPEG(联合摄影专家组)压缩标准,该标准被广泛用于数字摄影和图像编辑。


具有JPEG压缩功能的工业相机可以捕捉高分辨率的图像,并将其压缩成较小的文件,从而使图像的存储、传输和处理更加容易。压缩水平和质量可以根据用户的具体需求进行调整。


在工业相机中使用JPEG压缩的一个优点是,它可以帮助减少数据传输时间,这在需要高速图像处理的机器视觉应用中特别重要。此外,较小的文件大小允许更有效地存储和检索图像,这在工业环境中可以节省时间和金钱。


总的来说,JPEG图像压缩功能是工业相机的一个重要功能,有助于提高高分辨率图像的效率、存储和处理。

4.png

3.png

代码分析

Baumer工业相机堡盟相机SDK示例中022_JPEGCapture.cpp详细介绍了如何配置相机偏振功能。


软件SDK示例地址如下所示:Baumer_GAPI_SDK_2.12.0_win_x86_64_cpp\examples\src\0_Common\022_JPEGCapture\022_JPEGCapture.cpp


2.png


本例描述了对堡盟JPEG相机的处理。给出的源代码适用于处理一个系统、一台相机和十幅图像。


代码整体结构相对简单,在相机初始化后进行工业相机的JPEG图像压缩功能使用,部分核心代码如下:

std::cout << "DEVICE PARAMETER SETUP" << std::endl;
std::cout << "######################" << std::endl << std::endl;
try {
    // SET TRIGGER MODE OFF (FreeRun)
    pDevice->GetRemoteNode("TriggerMode")->SetString("Off");
    std::cout << "         TriggerMode:             "
        << pDevice->GetRemoteNode("TriggerMode")->GetValue() << std::endl;
    // JPEGTAG01 (see description on top)
    // check for JPEG capable camera
    if (pDevice->GetRemoteNodeList()->GetNodePresent(kImageCompressionMode.c_str())) {
        BGAPI2::NodeMap* image_compression_map =
            pDevice->GetRemoteNode(kImageCompressionMode.c_str())->GetEnumNodeList();
        if (image_compression_map->GetNodePresent(kImageCompressionModeJPEG.c_str())) {
            jpeg_streaming_capable = true;
            std::cout << "         JPEG Supported:          Yes" << std::endl;
        } else {
            std::cout << "         JPEG Supported:          No, Feature '" <<
                kImageCompressionMode << "' has no element " <<
                kImageCompressionModeJPEG <<
                std::endl;
        }
    } else {
        std::cout << "         JPEG Supported:          No Feature '" <<
            kImageCompressionMode << "' not found." <<
            std::endl;
    }
    if (jpeg_streaming_capable) {
        // JPEGTAG02 (see description on top)
        // AND SET PIXELFORMAT (Mono8, YCbCr) IF AVAILABLE
        BGAPI2::NodeMap * pixel_format_enum_map = pDevice->GetRemoteNode("PixelFormat")->GetEnumNodeList();
        std::string pixel_format = "YCbCr422_8";
        if (!pixel_format_enum_map->GetNodePresent(pixel_format.c_str()) || 
            (pixel_format_enum_map->GetNodePresent(pixel_format.c_str()) && !pixel_format_enum_map->GetNode(pixel_format.c_str())->IsReadable())) {
            pixel_format = "Mono8";
            if (!pixel_format_enum_map->GetNodePresent(pixel_format.c_str()) ||
                (pixel_format_enum_map->GetNodePresent(pixel_format.c_str()) && !pixel_format_enum_map->GetNode(pixel_format.c_str())->IsReadable())) {
                pixel_format = "";
            }
        }
        if(!pixel_format.empty() && pDevice->GetRemoteNode("PixelFormat")->IsWriteable()) {
            pDevice->GetRemoteNode("PixelFormat")->SetString(pixel_format.c_str());
            std::cout << "         PixelFormat:             "
                << pDevice->GetRemoteNode("PixelFormat")->GetValue() << std::endl;
            // JPEGTAG03 (see description on top)
            // set the compression mode to JPEG
            pDevice->GetRemoteNode(kImageCompressionMode.c_str())->SetValue("JPEG");
        } else {
            std::cout << "         PixelFormat:             "
                << pDevice->GetRemoteNode("PixelFormat")->GetValue() << " (set to Mono8/YCbCr failed, not supported or no write accees)" << std::endl;
        }
    }
    std::cout << std::endl;
}
catch (BGAPI2::Exceptions::IException& ex) {
    returncode = (returncode == 0) ? 1 : returncode;
    std::cout << "ExceptionType:    " << ex.GetType() << std::endl;
    std::cout << "ErrorDescription: " << ex.GetErrorDescription() << std::endl;
    std::cout << "in function:      " << ex.GetFunctionName() << std::endl;
}

下面功能为开启了JPEG图像压缩功能后,如何进行图像保存的功能方法,核心代码如下:

// CAPTURE 12 IMAGES
std::cout << " " << std::endl;
std::cout << "CAPTURE " << num_to_capture << " IMAGES BY IMAGE POLLING" << std::endl;
std::cout << "##################################" << std::endl << std::endl;
BGAPI2::Buffer * pBufferFilled = NULL;
try {
    // capture num_to_capture images
    for (unsigned int i = 0; i < num_to_capture; i++) {
        pBufferFilled = pDataStream->GetFilledBuffer(1000);  // timeout 1000 msec
        if (pBufferFilled == NULL) {
            std::cout << "Error: Buffer Timeout after 1000 msec" << std::endl;
        } else if (pBufferFilled->GetIsIncomplete() == true) {
            std::cout << "Error: Image is incomplete" << std::endl;
            pBufferFilled->QueueBuffer();
        } else {
            std::cout << " Image " << std::setw(5) << pBufferFilled->GetFrameID()
                << " received in memory address "
                << std::hex << pBufferFilled->GetMemPtr() << std::dec << std::endl;
            // JPEGTAG05 (see description on top)
            // save the compressed images to disk if supported
            if (jpeg_streaming_capable) {
                std::string jpeg_file = "022_JPEGCapture_";
                std::stringstream jpeg_filename;
                size_t jpeg_start_offset = pBufferFilled->GetImageOffset();
                size_t jpeg_end_offset = jpeg_start_offset + pBufferFilled->GetImageLength();
                if (jpeg_end_offset != 0) {
                    jpeg_filename.str("");
                    jpeg_filename << jpeg_file << std::setw(6) << std::setfill('0')
                        << pBufferFilled->GetFrameID() << ".jpg";
                    std::ofstream outfile;
                    outfile.open(jpeg_filename.str().c_str(), std::ios_base::binary);
                    outfile.write(
                        reinterpret_cast<const char*>(pBufferFilled->GetMemPtr()) + jpeg_start_offset,
                        jpeg_end_offset - jpeg_start_offset);
                    std::cout << "JPEG Data found. Image saved as JPEG file." << std::endl;
                } else {
                    std::cout << "JPEG Data not found in image buffer. No JPEG Saved." << std::endl;
                }
            }
            pBufferFilled->QueueBuffer();
        }
    }
}
catch (BGAPI2::Exceptions::IException& ex) {
    returncode = (returncode == 0) ? 1 : returncode;
    std::cout << "ExceptionType:    " << ex.GetType() << std::endl;
    std::cout << "ErrorDescription: " << ex.GetErrorDescription() << std::endl;
    std::cout << "in function:      " << ex.GetFunctionName() << std::endl;
}
std::cout << " " << std::endl;
std::cout << "CAMERA STOP" << std::endl;
std::cout << "###########" << std::endl << std::endl;

JPEG图像压缩功能的优势

1.JPEG

在好的压缩比下能够得到高质量的图片(例如 1:10)

被广泛接受的标准格式,通过MJPEG压缩得到AVI 视频成为可能

2.使处理过程变的简单

不PC软件压缩相比,降低了CPU的负载

在一台PC上可使用多台JPEG相机,有效降低成本

3.传输需要更低的带宽

在更低带宽接口上确保高分辨率和高速的数据传输

通过使用交换机支持远距离传输,有效降低成本

4.存档所需的存储空间更小

更长的存储时间,更低的存储介质成本

5.自由度更高,灵活性得以发挥

对应不同应用的需求更具有灵活性和自由度


JPEG图像压缩功能的行业应用


工业相机的JPEG图像压缩功能可广泛用于许多行业应用,如。


1. 监视。JPEG压缩功能可以为视频监控系统有效地存储和传输高质量的图像。


2. 医学成像。医疗成像系统经常使用 JPEG 压缩来存储和传输医疗图像。


3. 工业检查。工业检查系统可以使用 JPEG 压缩来存储生产过程中的缺陷或异常的图像。


4. 农业。在农业应用中,JPEG压缩可以被用来捕获和存储农作物和牲畜的图像。


5. 运输。在运输系统中,JPEG压缩可以用来存储和传输由放置在车辆上的摄像机所拍摄的图像,以保证安全和安保。


总的来说,JPEG图像压缩功能在许多工业应用中是一个必不可少的功能,因为它可以有效地存储和传输高质量的图像,提高生产力和减少运营成本。

目录
相关文章
|
14天前
|
存储 C++ UED
【实战指南】4步实现C++插件化编程,轻松实现功能定制与扩展
本文介绍了如何通过四步实现C++插件化编程,实现功能定制与扩展。主要内容包括引言、概述、需求分析、设计方案、详细设计、验证和总结。通过动态加载功能模块,实现软件的高度灵活性和可扩展性,支持快速定制和市场变化响应。具体步骤涉及配置文件构建、模块编译、动态库入口实现和主程序加载。验证部分展示了模块加载成功的日志和配置信息。总结中强调了插件化编程的优势及其在多个方面的应用。
139 61
|
3月前
|
算法框架/工具 C++ Python
根据相机旋转矩阵求解三个轴的旋转角/欧拉角/姿态角 或 旋转矩阵与欧拉角(Euler Angles)之间的相互转换,以及python和C++代码实现
根据相机旋转矩阵求解三个轴的旋转角/欧拉角/姿态角 或 旋转矩阵与欧拉角(Euler Angles)之间的相互转换,以及python和C++代码实现
148 0
|
23天前
|
存储 编译器 C++
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
42 2
|
2月前
|
编译器 C++
【C++核心】函数的应用和提高详解
这篇文章详细讲解了C++函数的定义、调用、值传递、常见样式、声明、分文件编写以及函数提高的内容,包括函数默认参数、占位参数、重载等高级用法。
21 3
|
3月前
|
存储 算法 C++
C++ STL应用宝典:高效处理数据的艺术与实战技巧大揭秘!
【8月更文挑战第22天】C++ STL(标准模板库)是一组高效的数据结构与算法集合,极大提升编程效率与代码可读性。它包括容器、迭代器、算法等组件。例如,统计文本中单词频率可用`std::map`和`std::ifstream`实现;对数据排序及找极值则可通过`std::vector`结合`std::sort`、`std::min/max_element`完成;而快速查找字符串则适合使用`std::set`配合其内置的`find`方法。这些示例展示了STL的强大功能,有助于编写简洁高效的代码。
43 2
|
2月前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
121 0
|
3月前
|
存储 编译器 C++
C++多态实现的原理:深入探索与实战应用
【8月更文挑战第21天】在C++的浩瀚宇宙中,多态性(Polymorphism)无疑是一颗璀璨的星辰,它赋予了程序高度的灵活性和可扩展性。多态允许我们通过基类指针或引用来调用派生类的成员函数,而具体调用哪个函数则取决于指针或引用所指向的对象的实际类型。本文将深入探讨C++多态实现的原理,并结合工作学习中的实际案例,分享其技术干货。
67 0
|
3月前
|
JSON Android开发 C++
Android c++ core guideline checker 应用
Android c++ core guideline checker 应用
|
9天前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
16 4
|
9天前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
15 4