opencv c++ 检测图像尺寸大小,标注轮廓

news2025/1/11 6:51:38

1. 项目背景

本项目旨在开发一个图像处理程序,通过使用计算机视觉技术,能够自动检测图像中物体的尺寸并进行分类。项目利用了开源的计算机视觉库 OpenCV,实现了图像的灰度处理、二值化、轮廓检测、边界框绘制以及尺寸分类等功能。通过这些功能,可以为用户提供一个便捷的工具,用于快速了解图像中物体的大小信息。

2. 技术与工具
  • 编程语言: C++
  • 主要库: OpenCV(版本1460)
  • 开发环境: Visual Studio(版本 2022),Windows 10
  • 版本控制: Git
3. 主要功能

本项目的主要功能包括:

  • 图像读取与预处理: 从文件系统中读取图像,并将其转换为灰度图像进行后续处理。
  • 二值化处理: 应用阈值将灰度图像转换为二值图像,以便进行轮廓检测。
  • 轮廓检测与筛选: 使用 OpenCV 提供的轮廓检测函数 findContours,并筛选出最大面积的轮廓。
  • 边界框绘制: 对检测到的最大面积轮廓绘制边界框,并计算其尺寸。
  • 尺寸分类: 根据边界框的尺寸(宽度和高度),将物体分为大、中、小三类,并输出分类结果。
  • 结果显示与保存: 将处理后的图像显示在窗口中,并可以选择保存处理结果。
4. 使用方法

用户可以通过以下步骤使用该项目:

  1. 准备图像: 将需要处理的图像放置在指定的目录中(例如 ../image/)。
  2. 运行程序: 在开发环境中编译并运行项目,或者直接运行已编译好的可执行文件。
  3. 查看结果: 程序将依次处理每张图像,检测物体的尺寸并输出分类结果。

#include <opencv2/opencv.hpp>
#include <iostream>
#include <vector>

using namespace std;
using namespace cv;

// 函数声明:处理单张图像并输出最大边界框尺寸类别
void processImage(const string& imagePath);

// 函数定义:处理单张图像并输出最大边界框尺寸类别
void processImage(const string& imagePath) {
    // 读取图像
    Mat image = imread(imagePath);

    // 检查图像是否成功读取
    if (image.empty()) {
        cout << "无法打开或找到图像: " << imagePath << endl;
        return; // 返回主函数继续处理下一张图像
    }

    // 将图像转换为灰度格式
    Mat img_gray;
    cvtColor(image, img_gray, COLOR_BGR2GRAY);

    // 应用二值化阈值处理

    int lower_gray_threshold = 35;  // 设置较低的灰度阈值 0
    int upper_gray_threshold = 90;  // 设置较高的灰度阈值 255

    Mat thresh;
    threshold(img_gray, thresh, lower_gray_threshold, upper_gray_threshold, THRESH_BINARY);

    // 在二值化图像上检测轮廓,使用 RETR_TREE 检索模式
    vector<vector<Point>> contours;
    vector<Vec4i> hierarchy;
    findContours(thresh, contours, hierarchy, RETR_TREE, CHAIN_APPROX_NONE);

    // 找到最大面积的轮廓
    double max_area = 0;
    int max_area_index = -1;
    for (size_t i = 0; i < contours.size(); i++) {
        double area = contourArea(contours[i]);
        if (area > max_area) {
            max_area = area;
            max_area_index = static_cast<int>(i);
        }
    }

    // 如果找到最大面积的轮廓,则绘制其边界框并输出尺寸类别
    if (max_area_index != -1) {
        Mat image_copy = image.clone();

        // 绘制最大面积轮廓
        drawContours(image_copy, contours, max_area_index, Scalar(0, 255, 0), 2);

        // 获取最大面积轮廓的边界框
        Rect bounding_rect = boundingRect(contours[max_area_index]);

        // 绘制边界框
        rectangle(image_copy, bounding_rect, Scalar(0, 0, 255), 2);

        // 获取边界框的中心点
        Point center(bounding_rect.x + bounding_rect.width / 2, bounding_rect.y + bounding_rect.height / 2);

        // 标注宽度和高度
        string text = "Width: " + to_string(bounding_rect.width) + ", Height: " + to_string(bounding_rect.height);
        int fontFace = FONT_HERSHEY_SIMPLEX;
        double fontScale = 0.5;
        int thickness = 1;
        int baseline = 0;
        Size textSize = getTextSize(text, fontFace, fontScale, thickness, &baseline);
        Point textOrg(center.x - textSize.width / 2, center.y + textSize.height / 2);
        putText(image_copy, text, textOrg, fontFace, fontScale, Scalar(255, 0, 0), thickness);

        // 输出边界框的尺寸
        int bounding_width = bounding_rect.width;
        int bounding_height = bounding_rect.height;
        string size_category;
        if (bounding_width >= 2000 && bounding_height >= 2000) {
            size_category = "大";
        }
        else if (bounding_width >= 1000 && bounding_height >= 1000) {
            size_category = "中";
        }
        else {
            size_category = "小";
        }

        cout << "图像: " << imagePath << ",尺寸:" << bounding_width << " x " << bounding_height << ",尺寸类别:" << size_category << endl;

        // 显示和保存结果(可选)
        // imshow("最大边界框", image_copy);
        // string output_filename = "largest_bounding_box_" + to_string(i) + ".jpg";
        // imwrite(output_filename, image_copy);
        // waitKey(0);
        // destroyAllWindows();
    }
    else {
        cout << "在图像 " << imagePath << " 中未找到符合条件的轮廓。" << endl;
    }
}

int main() {
    // 图像路径列表
    vector<string> imagePaths = {
        "D:/Project/image/001.jpg",
        "D:/Project/image/002.jpg",
        "D:/Project/image/003.jpg",
        "D:/Project/image/004.jpg",
        "D:/Project/image/005.jpg",
        "D:/Project/image/006.jpg",
        "D:/Project/image/007.jpg",
    };

    // 遍历处理每张图像
    for (const auto& imagePath : imagePaths) {
        processImage(imagePath);
    }

    return 0;
}

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

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

相关文章

Java控制语句——break和continue

系列文章目录 文章目录 系列文章目录前言一、【break的例子】二、【continue的例子】 前言 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站&#xff0c;这篇文章男女通用&#xff0c;看懂…

003 AOP介绍

文章目录 什么是AOP为什么用AOPAOP相关术语介绍AOP实现之AspectJ(了解)AOP实现之Spring AOP(了解)JDK动态代理Cglib动态代理 什么是AOP 在软件业中&#xff0c;AOP为Aspect Oriented Programming的缩写&#xff0c;意为面向切面编程作用&#xff1a;在不修改目标类代码的前提下…

Ubuntu乌班图安装VIM文本编辑器工具

系列文章目录 Ubuntu-24.04-live-server-amd64安装界面中文版 Ubuntu-24.04-live-server-amd64启用ssh Ubuntu安装qemu-guest-agent 文章目录 系列文章目录前言一、安装VIM&#xff1f;二、VIM基本设置总结 前言 从centos转到Ubuntu发现默认安装没有vi 一、安装VIM&#xff1…

服务器部署开源大模型完整教程 Ollama+Gemma+open-webui

现在开源的大模型其实挺多的&#xff0c;今天搭建Gemma这个谷歌开源的大模型&#xff0c;但是你想搭建别的只要你看完你都会了。 介绍 Ollama&#xff1a;一款可以让你在本地快速搭建大模型的工具 官网&#xff1a;https://ollama.com/ github&#xff1a;https://github.c…

【机器学习】智能创意工厂:机器学习驱动的AIGC,打造未来内容新生态

&#x1f680;时空传送门 &#x1f50d;机器学习在AIGC中的核心技术&#x1f4d5;深度学习&#x1f388;生成对抗网络&#xff08;GANs&#xff09; &#x1f680;机器学习在AIGC中的具体应用&#x1f340;图像生成与编辑⭐文本生成与对话系统&#x1f320;音频生成与语音合成 …

关于阿里云效流水线自动部署项目教程

1、登录阿里云效:阿里云登录 - 欢迎登录阿里云&#xff0c;安全稳定的云计算服务平台 2、点击左侧流水线&#xff1a; 3、在流水线界面&#xff0c;新建流水线 4、我的是php代码&#xff0c;因此选择php模版 5、创建之后添加流程线源&#xff0c;如下图 6、选择相应的源头。比…

【市场分析】TEMU平台美国区品牌产品市场分析Python采集爬虫数据

文章目录 一、引言二、数据概述1. 数据字段展示2. 数据统计信息展示 三、数据分析3.1 店铺托管模式3.1.1 半托管与全托管占比 3.2 品牌化最多的类目3.2.1 数据可视化展示3.2.2 各类目品牌商品占比分布 3.3 商品占比最多的Top品牌3.2.1 数据可视化展示3.2.2 品牌商品数量占比Top…

物联网技术-第6章-物联网应用案例

目录 1.共享单车 2.自动驾驶汽车 &#xff08;1&#xff09;概念 &#xff08;2&#xff09;关键技术 &#xff08;3&#xff09;典型代表 3.智能电网 4.智能交通 &#xff08;1&#xff09;车联网 &#xff08;2&#xff09;无人驾驶 5.智能物流 6.致谢 1.共享单车…

【语义分割系列】基于camvid数据集的Deeplabv3+分割算法(二)

基于camvid数据集的Deeplabv3+分割算法 前言 在前面的内容中,对比了Camvid数据集在基于不同backbone的Deeplabv3+算法上的效果。在这节内容中,本文将介绍在ghostnet的基础上,进一步优化效果,使得Miou提升。通过引入CFAC和CARAFE结构,有效地提升了模型的miou。 1.代码部…

JeecgFlow排他网关演示

排他网关概念理解 排他网关&#xff0c;也称为异或(XOR)网关&#xff0c;用于流程中实现分支决策建模。排他网关需要搭配条件顺序流使用。 当流程流转到排他网关时&#xff0c;所有流程顺序流都是会顺序求解&#xff0c; 其中第一条条件为true的顺序流会被选中(当有多条顺序流都…

每日一题——8行Python代码实现PAT乙级1029 旧键盘(举一反三+思想解读+逐步优化)五千字好文

一个认为一切根源都是“自己不够强”的INTJ 个人主页&#xff1a;用哲学编程-CSDN博客专栏&#xff1a;每日一题——举一反三Python编程学习Python内置函数 Python-3.12.0文档解读 目录 我的写法 代码评析 时间复杂度 空间复杂度 我要更强 方法一&#xff1a;使用集合…

OpenGL Super Bible 7th - Drawing Our First Triangle(绘制第一个三角形)

简介 本文的原版为《OpenGL Super Bible 7th》,是同事给我的,翻译是原文+译文的形势。文章不属于机器直译,原因在于语言不存在一一对应的关系,我将尽可能的按照中国人看起来舒服的方式来翻译这些段子,如果段子让你感到身心愉悦,那还劳烦点个关注,追个更。如果我没有及时…

IT入门知识第四部分《数据库》(4/10)

目录 1. 数据库基础 1.1 数据库的定义 1.2 数据库的关键概念 数据模型 数据库架构 数据库操作语言&#xff08;DML 和 DDL&#xff09; 总结 2. 关系型数据库 2.1 MySQL MySQL 的历史和特点 MySQL 的安装和配置 MySQL 的基本操作 2.2 PostgreSQL PostgreSQL 的特…

软件测试期末复习

软件测试期末复习 Author 雨 2024年6月18日 1. 什么是软件测试 从一个通常为无限的执行域中选取合适的有限的测试用例&#xff0c;对程序所期望的行为进行动态验证的活动过程。 2. 软件测试的目的 尽早地发现软件的缺陷 3.什么是测试什么是缺陷 从软件内部看&#xff1a;软件开…

在 Stable Diffusion 中控制光线的三种方式

光线在摄影中扮演着至关重要的角色&#xff0c;并对图像的整体质量和意境产生重要影响。你可以利用光线来增强主题&#xff0c;创造深度和立体感&#xff0c;传达情感&#xff0c;并突出重要细节。 在本文中&#xff0c;你将了解通过以下方法来控制光线&#xff1a; 光线提示…

【推荐100个unity插件之21】unity实现多语言切换功能——Localization插件的使用

文章目录 前言优缺点优点缺点 安装创建配置选择语言选择默认语言创建多语言表数据创建key配置不同语言文本预加载绑定不同多语言文本数据&#xff0c;并显示语言切换自己编写按钮控制语言切换多语言图片切换在构建中使用Localization分析错误修复动态修改多语言文本内容参考推荐…

高考后的抉择:如何在心仪专业与知名学校之间做出选择?

目录 前言1. 专业选择的深度探讨1.1 专业的优势与挑战1.1.1 课程学习1.1.2 就业前景 1.2 专业选择的个人经验与思考 2. 名校对个人发展的长短期影响2.1 名校声誉的品牌效应2.1.1 职业发展2.1.2 社会认可度 2.2 教育资源与学术氛围2.2.1 教育资源2.2.2 学术氛围 2.3 就业优势 3.…

【2024最新精简版】RabbitMQ面试篇

文章目录 Kafka和RabbitMQ什么区别惰性队列&#xff08;Lazy Queues&#xff09;是怎么实现的&#xff1f;RabbitMQ工作模式有哪些 ?你们项目中哪里用到了RabbitMQ ?为什么会选择使用RabbitMQ ? 有什么好处 ?使用RabbitMQ如何保证消息不丢失 ?消息的重复消费问题如何解决的…

3.1、前端异步编程(超详细手写实现Promise;实现all、race、allSettled、any;async/await的使用)

前端异步编程规范 Promise介绍手写Promise&#xff08;resolve&#xff0c;reject&#xff09;手写Promise&#xff08;then&#xff09;Promise相关 API实现allraceallSettledany async/await和Promise的关系async/await的使用 Promise介绍 Promise是一个类&#xff0c;可以翻…

Spring技术——介绍、初识

从这篇博客正式开始学习 Spring 。 一、整个Spring技术学习的介绍 首先先对 Spring 做一个简单的认识&#xff0c;并从以下3个方面了解学习Spring 技术的意义&#xff1a; &#xff08;1&#xff09;为什么要学习 Spring 技术&#xff1f; 1、它的市场的占有率很高&#xff…