OpenCV计算机视觉开发实践:基于Qt C++ - 商品搜索 - 京东

OpenCV中除了提供绘制各种图形的函数外,还提供了一个特殊的绘制函数,用于在图像上绘制文字。这个函数是putText(),它是命名空间cv中的函数,其声明如下:

void cv::putText(cv::Mat& img,          // 待绘制的图像const string& text,    // 待绘制的文字cv::Point origin,      // 文本框的左下角int fontFace,          // 字体 (如cv::FONT_HERSHEY_PLAIN)double fontScale,      // 尺寸因子,值越大,文字就越大cv::Scalar color,      // 线条的颜色(RGB)int thickness = 1,     // 线条宽度int lineType = 8,      // 线型(4邻域或8邻域,默认为8邻域)bool bottomLeftOrigin = false // true='origin at lower left'
);

这个函数可以简单地在图像上绘制一些文字,由text指定的文字将在以左上角为原点的文字框中以color指定的颜色绘制出来,除非bottomLeftOrigin标志设置为真,这种情况以左下角为原点,使用的字体由fontFace参数决定。常用的字体宏是FONT_HERSHEY_SIMPLEX(普通大小无衬线字体)和FONT_HERSHEY_PLAIN(小号无衬线字体)。任何一个字体都可以和CV::FONT_ITALIC 组合使用(通过“或”操作,或操作符号是|)来得到斜体。每种字体都有一个“自然”大小,当fontScale不是1.0时,在文字绘制之前字体大小将由这个数来缩放。

这里解释一下衬线。衬线指的是字母结构笔画之外的装饰性笔画。有衬线的字体叫衬线体(Serif),没有衬线的字体叫无衬线体(Sans-Serif)。衬线体的特征是在字的笔画开始、结束的地方有额外的装饰,而且笔画的粗细会有所不同。衬线体很容易识别,它强调了每个字母笔画的开始和结束,因此易读性比较高。中文字体中的宋体就是一种标准的衬线体。无衬线体(Sans-Serif Font)没有额外的装饰,而且笔画的粗细差不多。这类字体通常是机械的和统一线条的,它们往往拥有相同的曲率、笔直的线条和锐利的转角。无衬线体与汉字字体中的黑体相对应。

另外,在实际绘制文字之前,还可以使用cv::getTextSize()接口先获取待绘制文本框的大小,以方便放置文本框。getTextSize函数可以获取字符串的宽度和高度,该函数声明如下:

Size cv::getTextSize(const string& text,cv::Point origin,int fontFace,double fontScale,int thickness,int* baseLine);

其中参数text表示输入的文本文字;fontFace表示文字字体类型;fontScale表示字体缩放系数;thickness表示字体笔画线宽;baseLine是一个输出参数,表示文字最底部的y坐标。函数返回值中包含文本框的大小。

【例4.12】绘制文字

   新建一个控制台工程,工程名是test。

   打开main.cpp,输入如下代码:

#include <iostream>
#include <opencv2/core.hpp>
#include <opencv2/highgui.hpp>
#include<opencv2\imgproc.hpp>
using namespace std;
using namespace cv;int main()
{string text = "Funny text inside the box";// int fontFace = FONT_HERSHEY_SCRIPT_SIMPLEX;     // 手写风格字体int fontFace = FONT_HERSHEY_SCRIPT_COMPLEX;double fontScale = 2;       // 字体缩放比int thickness = 3;Mat img(600, 800, CV_8UC3, Scalar::all(0));int baseline = 0;Size textSize = getTextSize(text, fontFace, fontScale, thickness, &baseline);baseline += thickness;// center the textPoint textOrg((img.cols - textSize.width) / 2, (img.rows - textSize.height) / 2);// draw the boxrectangle(img, textOrg + Point(0, baseline), textOrg + Point(textSize.width, -textSize.height), Scalar(0, 0, 255));line(img, textOrg + Point(0, thickness), textOrg + Point(textSize.width, thickness), Scalar(0, 0, 255));putText(img, text, textOrg, fontFace, fontScale, Scalar::all(255), thickness, 8);imshow("text", img);waitKey(0);return 0;
}

在上述代码中,我们通过getTextSize函数获取包含字体的文本框的大小,并画线显示在矩形中。最后通过文本绘制函数putText画出一段字符串“Funny text inside the box”。

   保存工程并运行,结果如图4-8所示。

图4-8

下面我们通过一个比较综合的例子来实现随机画图,比如随机画线,随机画圆,随机绘制文本等。

【例4.13】综合实例:随机画图

   新建一个控制台工程,工程名是test。

   打开main.cpp,输入如下代码:

#include <opencv2/opencv.hpp>
using namespace cv;
#include<iostream>
using namespace std;
#define FALSE 0
#define TRUE 1#include <opencv2/core.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
#include <iostream>
#include <stdio.h>
using namespace cv;
const int NUMBER = 100;
const int DELAY = 5;
const int window_width = 900;
const int window_height = 600;
int x_1 = -window_width / 2;
int x_2 = window_width * 3 / 2;
int y_1 = -window_width / 2;
int y_2 = window_width * 3 / 2;
static Scalar randomColor(RNG& rng);
int Drawing_Random_Lines(Mat image, char* window_name, RNG rng);
int Drawing_Random_Rectangles(Mat image, char* window_name, RNG rng);
int Drawing_Random_Ellipses(Mat image, char* window_name, RNG rng);
int Drawing_Random_Polylines(Mat image, char* window_name, RNG rng);
int Drawing_Random_Filled_Polygons(Mat image, char* window_name, RNG rng);
int Drawing_Random_Circles(Mat image, char* window_name, RNG rng);
int Displaying_Random_Text(Mat image, char* window_name, RNG rng);
int Displaying_Big_End(Mat image, char* window_name, RNG rng);
int main(void)
{int c;char window_name[] = "Drawing_2 Tutorial";RNG rng(0xFFFFFFFF);Mat image = Mat::zeros(window_height, window_width, CV_8UC3);  // 创建一个初始化为零的矩阵imshow(window_name, image);waitKey(DELAY);// 然后我们开始随机画图c = Drawing_Random_Lines(image, window_name, rng);if (c != 0) return 0;c = Drawing_Random_Rectangles(image, window_name, rng);if (c != 0) return 0;c = Drawing_Random_Ellipses(image, window_name, rng);if (c != 0) return 0;c = Drawing_Random_Polylines(image, window_name, rng);if (c != 0) return 0;c = Drawing_Random_Filled_Polygons(image, window_name, rng);if (c != 0) return 0;c = Drawing_Random_Circles(image, window_name, rng);if (c != 0) return 0;c = Displaying_Random_Text(image, window_name, rng);if (c != 0) return 0;c = Displaying_Big_End(image, window_name, rng);if (c != 0) return 0;waitKey(0);return 0;
}
static Scalar randomColor(RNG& rng)
{int icolor = (unsigned)rng;return Scalar(icolor & 255, (icolor >> 8) & 255, (icolor >> 16) & 255);
}
int Drawing_Random_Lines(Mat image, char* window_name, RNG rng)
{Point pt1, pt2;for (int i = 0; i < NUMBER; i++){pt1.x = rng.uniform(x_1, x_2);pt1.y = rng.uniform(y_1, y_2);pt2.x = rng.uniform(x_1, x_2);pt2.y = rng.uniform(y_1, y_2);line(image, pt1, pt2, randomColor(rng), rng.uniform(1, 10), 8);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Drawing_Random_Rectangles(Mat image, char* window_name, RNG rng)
{Point pt1, pt2;int lineType = 8;int thickness = rng.uniform(-3, 10);for (int i = 0; i < NUMBER; i++){pt1.x = rng.uniform(x_1, x_2);pt1.y = rng.uniform(y_1, y_2);pt2.x = rng.uniform(x_1, x_2);pt2.y = rng.uniform(y_1, y_2);rectangle(image, pt1, pt2, randomColor(rng), MAX(thickness, -1), lineType);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Drawing_Random_Ellipses(Mat image, char* window_name, RNG rng)
{int lineType = 8;for (int i = 0; i < NUMBER; i++){Point center;center.x = rng.uniform(x_1, x_2);center.y = rng.uniform(y_1, y_2);Size axes;axes.width = rng.uniform(0, 200);axes.height = rng.uniform(0, 200);double angle = rng.uniform(0, 180);ellipse(image, center, axes, angle, angle - 100, angle + 200,randomColor(rng), rng.uniform(-1, 9), lineType);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Drawing_Random_Polylines(Mat image, char* window_name, RNG rng)
{int lineType = 8;for (int i = 0; i < NUMBER; i++){Point pt[2][3];pt[0][0].x = rng.uniform(x_1, x_2);pt[0][0].y = rng.uniform(y_1, y_2);pt[0][1].x = rng.uniform(x_1, x_2);pt[0][1].y = rng.uniform(y_1, y_2);pt[0][2].x = rng.uniform(x_1, x_2);pt[0][2].y = rng.uniform(y_1, y_2);pt[1][0].x = rng.uniform(x_1, x_2);pt[1][0].y = rng.uniform(y_1, y_2);pt[1][1].x = rng.uniform(x_1, x_2);pt[1][1].y = rng.uniform(y_1, y_2);pt[1][2].x = rng.uniform(x_1, x_2);pt[1][2].y = rng.uniform(y_1, y_2);const Point* ppt[2] = { pt[0], pt[1] };int npt[] = { 3, 3 };polylines(image, ppt, npt, 2, true, randomColor(rng), rng.uniform(1, 10), lineType);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Drawing_Random_Filled_Polygons(Mat image, char* window_name, RNG rng)
{int lineType = 8;for (int i = 0; i < NUMBER; i++){Point pt[2][3];pt[0][0].x = rng.uniform(x_1, x_2);pt[0][0].y = rng.uniform(y_1, y_2);pt[0][1].x = rng.uniform(x_1, x_2);pt[0][1].y = rng.uniform(y_1, y_2);pt[0][2].x = rng.uniform(x_1, x_2);pt[0][2].y = rng.uniform(y_1, y_2);pt[1][0].x = rng.uniform(x_1, x_2);pt[1][0].y = rng.uniform(y_1, y_2);pt[1][1].x = rng.uniform(x_1, x_2);pt[1][1].y = rng.uniform(y_1, y_2);pt[1][2].x = rng.uniform(x_1, x_2);pt[1][2].y = rng.uniform(y_1, y_2);const Point* ppt[2] = { pt[0], pt[1] };int npt[] = { 3, 3 };fillPoly(image, ppt, npt, 2, randomColor(rng), lineType);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Drawing_Random_Circles(Mat image, char* window_name, RNG rng)
{int lineType = 8;for (int i = 0; i < NUMBER; i++){Point center;center.x = rng.uniform(x_1, x_2);center.y = rng.uniform(y_1, y_2);circle(image, center, rng.uniform(0, 300), randomColor(rng),rng.uniform(-1, 9), lineType);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Displaying_Random_Text(Mat image, char* window_name, RNG rng)
{int lineType = 8;for (int i = 1; i < NUMBER; i++){Point org;org.x = rng.uniform(x_1, x_2);org.y = rng.uniform(y_1, y_2);putText(image, "Testing text rendering", org, rng.uniform(0, 8),rng.uniform(0, 100)*0.05 + 0.1, randomColor(rng), rng.uniform(1, 10), lineType);imshow(window_name, image);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}
int Displaying_Big_End(Mat image, char* window_name, RNG)
{Size textsize = getTextSize("OpenCV forever!", FONT_HERSHEY_COMPLEX, 3, 5, 0);Point org((window_width - textsize.width) / 2, (window_height - textsize.height) / 2);int lineType = 8;Mat image2;for (int i = 0; i < 255; i += 2){image2 = image - Scalar::all(i);putText(image2, "OpenCV forever!", org, FONT_HERSHEY_COMPLEX, 3,Scalar(i, i, 255), 5, lineType);imshow(window_name, image2);if (waitKey(DELAY) >= 0){return -1;}}return 0;
}

从上面代码可以看到,在主函数中,要做的第一件事是创建一个随机数生成器对象:RNG rng(0xFFFFFFFF);。在本例中,RNG使用值0xFFFFFFFF来进行初始化。

然后创建一个初始化为零的矩阵image(这意味着它将显示为黑色),并指定其高度、宽度和类型。

接着开始随机画图。查看代码可以看到,这里主要是8个自定义函数。这些函数都遵循相同的模式,因此我们只分析其中几个函数,因为相同的解释适用于所有函数。比如函数Drawing_Random_Lines用来在随机坐标处画线,因此画线函数line的首尾点pt1和pt2的坐标都是随机生成的。同样地,函数Drawing_Random_Circles用来随机画圆;函数Displaying_Random_Text用来绘制文本,并且文本的位置、颜色等也是随机产生的;函数Displaying_Big_End用来绘制程序结束时所显示的文本“OpenCV forever!”。

   保存工程并运行,结果如图4-9所示。

图4-9

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

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

synchronized的技巧与要点

一、基本概念 目的&#xff1a;解决多线程并发访问共享资源时的数据竞争问题&#xff0c;保证原子性、可见性和有序性&#xff08;JMM内存模型&#xff09;。性质&#xff1a;可重入锁&#xff08;同一线程可重复获取同一把锁&#xff09;、独占锁&#xff08;互斥锁&#xff…

特殊混淆案例还原指南:突破变形控制流与量子加密的技术解析

引言​​ 在JavaScript混淆领域,传统的字符串加密和控制流平坦化已无法满足高端防护需求。2023年Snyk安全报告指出,Top级商业产品已转向​​多态变形控制流​​和​​量子加密技术​​,这类混淆方案占比17%,但导致的反向工程失败率高达94%。本文将通过三个工业级混淆案例(…

基于Python、tkinter、sqlite3 和matplotlib的校园书店管理系统

写一个小例子练习一下python语言。一个基于Python的校园书店管理系统&#xff0c;使用了tkinter库构建图形用户界面&#xff08;GUI&#xff09;&#xff0c;sqlite3 进行数据库管理&#xff0c;matplotlib用于统计分析可视化。系统支持用户登录、书籍管理、客户管理、员工管理…

机器学习×第十四卷:集成学习中篇——她从每次错误中修正自己

&#x1f380;【开场 她终于愿意回看自己贴错的地方了】 &#x1f98a;狐狐&#xff1a;“她贴过你很多次&#xff0c;但每次贴歪了&#xff0c;都只是低头沉默。” &#x1f43e;猫猫&#xff1a;“现在不一样了喵……她开始反思&#xff0c;是不是哪一刻该绕过来贴你背后&…

LeetCode 2537.统计好子数组的数目

给你一个整数数组 nums 和一个整数 k &#xff0c;请你返回 nums 中 好 子数组的数目。 一个子数组 arr 如果有 至少 k 对下标 (i, j) 满足 i < j 且 arr[i] arr[j] &#xff0c;那么称它是一个 好 子数组。 子数组 是原数组中一段连续 非空 的元素序列。 示例 1&#x…

Python 开发环境管理和常用命令

包管理器选择 从轻到重: venv → virtualenv → conda venv: Python 3.3 内置&#xff0c;轻量级虚拟环境virtualenv: 第三方包&#xff0c;支持更多Python版本conda: 科学计算友好&#xff0c;包含包管理和环境管理 Python 版本支持 查看各版本支持状态&#xff1a;Status…

macOS - 根据序列号查看机型、保障信息

文章目录 最近在看 MacBook 二手机&#xff0c;有个咸鱼卖家放个截图 说不清参数&#xff0c;于是想根据 序列号 查看机型。苹果提供了这样的网页&#xff1a; https://checkcoverage.apple.com/ &#xff08;无需登录&#xff09; 结果 2025-06-20&#xff08;五&#xff09;…

数字化项目调研过程中需要的文章

Hello&#xff0c;大家好 &#xff0c;我是东哥说-MES 最近写了不少的文章&#xff0c;为了方便阅读&#xff0c;特意重新整理了数字化相关的文章链接&#xff0c;也便于大家阅读 数字工厂项目启动与业务需求调研执行指南-CSDN博客文章浏览阅读725次&#xff0c;点赞28次&…

LangChain4j之会话功能AiServices工具类的使用(系列二)

概述 LangChain4j 的会话功能 AiServices 工具类&#xff0c;可助力轻松实现智能对话。它能记录对话上下文&#xff0c;让 AI 回答连贯且贴合情境。使用起来&#xff0c;先引入类&#xff0c;配置好相关参数&#xff0c;如模型地址、密钥等。接着&#xff0c;调用相应方法传入…

Qt 中使用 gtest 做单元测试

作者&#xff1a;billy 版权声明&#xff1a;著作权归作者所有&#xff0c;商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处 gtest 简介 GoogleTest&#xff08;也称为gtest&#xff09;是由 Google 开发的一个 C 单元测试框架&#xff0c;用于编写、组织和运行…

WPF TreeView 数据绑定完全指南:MVVM 模式实现

WPF TreeView 数据绑定完全指南&#xff1a;MVVM 模式实现 一、TreeView 绑定的核心概念1.1 MVVM 模式下的 TreeView 绑定原理1.2 绑定关系示意图 二、完整实现步骤2.1 创建节点模型类2.2 创建 ViewModel2.3 XAML 绑定配置2.4 设置 Window 的 DataContext 三、关键特性详解3.1 …

坤驰科技QTS4200战鹰(Battle Eagle)系列实时频谱分析记录回放系统

QTS4200战鹰(Battle Eagle)系列 实时频谱分析记录回放系统 精准采集&#xff5c;高效回放&#xff5c;拓展频谱分析新边界 坤驰科技倾力打造新一代集实时频谱分析、高速信号记录与精准信号回放于一体的便携式系统&#xff0c;为无线电监测、无线通信、国防等领域提供全面而高…

Flask(二) 路由routes

文章目录 基本路由定义路由参数路由规则设置请求方法&#xff08;GET/POST&#xff09;路由函数返回静态文件和模板Blueprint&#xff08;模块化路由&#xff09;显示当前所有路由 Flask 路由是 Web 应用程序中将 URL 映射到 Python 函数的机制。 定义路由&#xff1a;使用 app…

在el-image组件的预览中添加打印功能(自定义功能)

思路&#xff1a;给图片添加点击事件&#xff0c;通过js获取预览的工具栏&#xff0c;在工具栏中添加自定义按钮及事件 1、html 中 image标签 <el-image style"width: 139px; height: 89px" :src"fileUrl" :preview-src-list"[fileUrl]" cli…

TongWeb替换tomcat

1、背景 国家近年来大力推进信息技术应用创新产业&#xff08;信创&#xff09;&#xff0c;要求关键领域采用自主可控的国产软硬件。Tomcat作为国外开源产品&#xff0c;存在潜在的安全风险和技术依赖。TongWeb作为国产中间件&#xff0c;符合信创目录要求&#xff0c;满足政府…

联合语音和文本机器翻译,支持多达100种语言(nature子刊论文研读)

简介&#xff1a; &#x1f30d; SEAMLESSM4T 是一种单一模型&#xff0c;实现了跨越多达 101 种源语言和多种目标语言的语音到语音、语音到文本、文本到语音和文本到文本翻译及自动语音识别。&#x1f680; 该模型性能显著超越现有级联系统&#xff0c;特别是在语音到文本和语…

网站公安网安备案查询API集成指南

网站公安网安备案查询API集成指南 引言 随着互联网应用的日益普及&#xff0c;网络安全和个人隐私保护越来越受到重视。公安网安备案作为保障网络安全的重要措施之一&#xff0c;对于确保网站合法合规运营具有重要意义。为了帮助开发者更加便捷地获取网站的公安网安备案信息&a…

如何用远程调试工具排查 WebView 与原生通信问题(iOS或Android)

WebView 在移动端开发中的角色越来越关键&#xff0c;尤其在混合架构&#xff08;Hybrid&#xff09;项目中&#xff0c;它作为前端与原生的桥梁&#xff0c;承载了大量交互行为。但这个桥梁并不总是稳固&#xff0c;尤其是在涉及 JSBridge 通信 时&#xff0c;前端调用原生接口…

使用 spark-submit 运行依赖第三方库的 Python 文件

python文件在spark集群运行真的麻烦&#xff0c;烦冗 spark运行分为了三个模式&#xff0c;本地模式/client模式/cluster模式 文章目录 本地模式client模式cluster模式参考 本地模式 现在的spark支持python3了&#xff0c;支持python2的版本已经很落后了&#xff0c;所以需要…

【android bluetooth 协议分析 05】【蓝牙连接详解2】【acl_interface_t介绍】

1. acl_interface_t 介绍 acl_interface_t 结构体及其子结构体&#xff0c;目的是封装处理 Classic、LE、SCO 连接及链路事件的回调函数&#xff0c;用于 HCI 事件与上层蓝牙协议栈的解耦分发。 system/main/shim/acl_legacy_interface.h typedef struct {void (*on_connect…