34 Opencv 自定义角点检测

news/2024/12/23 12:29:56 标签: opencv, 人工智能, 计算机视觉

文章目录

  • cornerEigenValsAndVecs
  • cornerMinEigenVal
  • 示例

cornerEigenValsAndVecs

void cornerEigenValsAndVecs(
       InputArray src,       --单通道输入8位或浮点图像
       OutputArray dst,    --输出图像,同源图像或CV_32FC(6)
       int blockSize,         --邻域大小值
       int apertureSize,    --Sobel算子当中的核大小
       int borderType=BORDER_DEFAULT --像素外插方法
)//对应于Harris

cornerMinEigenVal

void cornerMinEigenVal(
       InputArray src,     --单通道输入8位或浮点图像
       OutputArray dst,  --图像存储的最小特征值。类型为CV_32FC1
       int blockSize,       --邻域大小值
       int apertureSize=3,   --Sobel算子的参数
       int borderType=BORDER_DEFAULT  --像素外插方法
}//对应Shi-Tomasi

示例

#include <opencv2/opencv.hpp> // 包含OpenCV库的头文件
#include <iostream>           // 包含标准输入输出流库

#include <math.h>             // 包含数学函数库
using namespace cv;           // 使用cv命名空间,以避免每次调用OpenCV函数时都要写cv::
using namespace std;          // 使用std命名空间,以便使用cout, cin等函数

const char* harris_win = "Custom Harris Corners Detector"; // 定义Harris角点检测结果窗口名称
const char* shitomasi_win = "Custom Shi-Tomasi Corners Detector"; // 定义Shi-Tomasi角点检测结果窗口名称
Mat src, gray_src;            // 声明源图像和灰度图变量

// Harris角点响应相关的矩阵及最大最小响应值
Mat harris_dst, harrisRspImg;
double harris_min_rsp;
double harris_max_rsp;

// Shi-Tomasi角点响应相关的矩阵及最大最小响应值
Mat shiTomasiRsp;
double shitomasi_max_rsp;
double shitomasi_min_rsp;

int sm_qualitylevel = 30;     // Shi-Tomasi质量级别初始值
int qualityLevel = 30;        // Harris质量级别初始值
int max_count = 100;          // 滑动条的最大值

void CustomHarris_Demo(int, void*); // 声明Harris角点检测回调函数
void CustomShiTomasi_Demo(int, void*);// 声明Shi-Tomasi角点检测回调函数

int main(int argc, char** argv) {
    src = imread("D:/vcprojects/images/home.jpg"); // 读取图片到src Mat对象中
    if (src.empty()) {                             // 检查是否成功加载图片
        printf("could not load image...\n");
        return -1;
    }
    namedWindow("input image", CV_WINDOW_AUTOSIZE); // 创建一个名为"input image"的窗口
    imshow("input image", src);                     // 在该窗口中显示原始图像

    cvtColor(src, gray_src, COLOR_BGR2GRAY);        // 将BGR格式的src转换为灰度图gray_src

    // 计算特征值(Harris角点)
    int blockSize = 3;      // 邻域大小
    int ksize = 3;          // Sobel算子的孔径参数
    double k = 0.04;        // Harris公式中的自由参数k
    harris_dst = Mat::zeros(src.size(), CV_32FC(6)); // 初始化用于存储特征值和向量的矩阵
    harrisRspImg = Mat::zeros(src.size(), CV_32FC1); // 初始化用于存储Harris响应值的矩阵
    cornerEigenValsAndVecs(gray_src, harris_dst, blockSize, ksize, 4); // 计算每个像素处的特征值和特征向量

    // 计算Harris响应值
    for (int row = 0; row < harris_dst.rows; row++) {
        for (int col = 0; col < harris_dst.cols; col++) {
            double lambda1 = harris_dst.at<Vec6f>(row, col)[0]; // 获取第一个特征值
            double lambda2 = harris_dst.at<Vec6f>(row, col)[1]; // 获取第二个特征值
            harrisRspImg.at<float>(row, col) = lambda1 * lambda2 - k * pow((lambda1 + lambda2), 2); // 计算Harris响应值
        }
    }

    minMaxLoc(harrisRspImg, &harris_min_rsp, &harris_max_rsp, 0, 0, Mat()); // 找到响应值中的最大最小值

    namedWindow(harris_win, CV_WINDOW_AUTOSIZE); // 创建一个用于显示Harris角点检测结果的窗口
    createTrackbar("Quality Value:", harris_win, &qualityLevel, max_count, CustomHarris_Demo); // 创建滑块以调整质量级别
    CustomHarris_Demo(0, 0); // 初始调用回调函数

    // 计算最小特征值(Shi-Tomasi角点)
    shiTomasiRsp = Mat::zeros(src.size(), CV_32FC1); // 初始化用于存储Shi-Tomasi响应值的矩阵
    cornerMinEigenVal(gray_src, shiTomasiRsp, blockSize, ksize, 4); // 计算每个像素处的最小特征值

    minMaxLoc(shiTomasiRsp, &shitomasi_min_rsp, &shitomasi_max_rsp, 0, 0, Mat()); // 找到响应值中的最大最小值

    namedWindow(shitomasi_win, CV_WINDOW_AUTOSIZE); // 创建一个用于显示Shi-Tomasi角点检测结果的窗口
    createTrackbar("Quality:", shitomasi_win, &sm_qualitylevel, max_count, CustomShiTomasi_Demo); // 创建滑块以调整质量级别
    CustomShiTomasi_Demo(0, 0); // 初始调用回调函数

    waitKey(0); // 等待按键事件
    return 0;
}

// Harris角点检测回调函数:根据用户设定的质量级别绘制角点
void CustomHarris_Demo(int, void*) {
    if (qualityLevel < 10) { // 设定最低质量级别
        qualityLevel = 10;
    }
    Mat resultImg = src.clone(); // 复制原始图像用于绘制角点
    float t = harris_min_rsp + (((double)qualityLevel) / max_count) * (harris_max_rsp - harris_min_rsp); // 根据质量级别计算阈值
    for (int row = 0; row < src.rows; row++) {
        for (int col = 0; col < src.cols; col++) {
            float v = harrisRspImg.at<float>(row, col); // 获取当前像素的Harris响应值
            if (v > t) { // 如果响应值大于阈值,则认为是角点
                circle(resultImg, Point(col, row), 2, Scalar(0, 0, 255), 2, 8, 0); // 在图像上绘制红色圆圈标记角点
            }
        }
    }
    imshow(harris_win, resultImg); // 显示带有角点标记的图像
}

// Shi-Tomasi角点检测回调函数:根据用户设定的质量级别绘制角点
void CustomShiTomasi_Demo(int, void*) {
    if (sm_qualitylevel < 20) { // 设定最低质量级别
        sm_qualitylevel = 20;
    }

    Mat resultImg = src.clone(); // 复制原始图像用于绘制角点
    float t = shitomasi_min_rsp + (((double)sm_qualitylevel) / max_count) * (shitomasi_max_rsp - shitomasi_min_rsp); // 根据质量级别计算阈值
    for (int row = 0; row < src.rows; row++) {
        for (int col = 0; col < src.cols; col++) {
            float v = shiTomasiRsp.at<float>(row, col); // 获取当前像素的Shi-Tomasi响应值
            if (v > t) { // 如果响应值大于阈值,则认为是角点
                circle(resultImg, Point(col, row), 2, Scalar(0, 0, 255), 2, 8, 0); // 在图像上绘制红色圆圈标记角点
            }
        }
    }
    imshow(shitomasi_win, resultImg); // 显示带有角点标记的图像
}

http://www.niftyadmin.cn/n/5796599.html

相关文章

3354. 使数组元素等于零

3354、[简单] 使数组元素等于零 1、题目描述 给你一个整数数组 nums 。 开始时&#xff0c;选择一个满足 nums[curr] 0 的起始位置 curr &#xff0c;并选择一个移动 方向 &#xff1a;向左或者向右。 此后&#xff0c;你需要重复下面的过程&#xff1a; 如果 curr 超过范…

如何使用Selenium处理JavaScript动态加载的内容?

在现代Web开发中&#xff0c;JavaScript已经成为实现动态内容和交互的核心技术。对于爬虫开发者来说&#xff0c;处理JavaScript动态加载的内容是一个常见的挑战。Selenium是一个强大的工具&#xff0c;它可以模拟真实用户的浏览器行为&#xff0c;从而获取完整的页面内容。本文…

Linux文件目录 --- 复制命令CP、递归复制目录、软连接、硬链接

一、复制cp 该命令用于复制文件或目录&#xff0c;下面是命令使用格式和常用的参数 cp [参数] 源文件或目录 目标文件或目录 #中间各有一个空格隔开 参数作用-f覆盖同名文件或目录时不进行提醒-i …

设计模式の命令访问者迭代器模式

文章目录 前言一、命令模式二、访问者模式三、迭代器模式 前言 本篇是关于设计模式中命令模式、访问者模式、以及迭代器模式的学习笔记。 一、命令模式 命令模式是一种行为型设计模式&#xff0c;其核心目的在于将命令的发送者和接受者解耦&#xff0c;提供一个中间层对命令进行…

计算机网络 - HTTP 协议和万维网

基本概念 万维网 (World Wide Web, WWW) 定义&#xff1a;一个大规模的分布式信息系统&#xff0c;由全球范围内无数个网络站点和网页组成特点&#xff1a;基于超文本技术&#xff0c;支持多媒体内容的展示和交互URL (Uniform Resource Locator) 定义&#xff1a;统一资源定位…

【学习总结|DAY022】Java网络编程

网络编程是Java开发中非常重要的一环&#xff0c;它允许程序与网络上的其他设备进行数据交互。本文将介绍Java网络编程的基础知识&#xff0c;包括网络编程三要素、UDP和TCP通信协议&#xff0c;以及BS架构的原理。 网络编程三要素 网络通信至少需要三个要素&#xff1a;IP地…

SAP-ABAP开发学习-面向对象开发ooalv(2)

SAP-ABAP开发学习-面向对象OOALV&#xff08;1&#xff09;-CSDN博客 本文目录 一、类的继承 多态性类继承的实现 二、抽象类 三、最终类 四、接口 五、定义全局对象 一、类的继承 继承的本质是代码重用。当我们要构造一个新类时&#xff0c;无需从零开始&#xff0c;可…

微调 BERT:实现抽取式问答

学习如何使用 Transformers 库微调预训练模型来实现抽取式问答。 本文的思路与 08. 尝试微调 LLM&#xff1a;让它会写唐诗一致&#xff0c;不过这次我们使用 BERT 作为预训练模型进行演示&#xff0c;并进行全量微调而非使用 LoRA。为了更好地解释细节&#xff0c;我们不使用 …