使用 C/C++ 和 OpenCV 实现滑动条控制图像旋转

使用 C++ 和 OpenCV 实现滑动条控制图像旋转

本文将介绍如何使用 C++ 和 OpenCV 库创建一个简单的应用程序,该程序可以显示一张图片,并允许用户通过一个滑动条(Trackbar)来实时控制图片的旋转角度。这是一个非常实用的交互式功能,能帮助我们直观地理解仿射变换。

核心函数

在实现这个功能之前,我们主要会用到以下几个 OpenCV 的核心函数:

  1. namedWindow(): 创建一个可以容纳图像和滑动条的窗口。
  2. createTrackbar(): 在指定的窗口中创建一个滑动条。
  3. getRotationMatrix2D(): 计算二维旋转的仿射变换矩阵。它需要旋转中心、旋转角度和缩放比例作为参数。
  4. warpAffine() : 对图像应用一个仿射变换。我们将使用 getRotationMatrix2D() 生成的矩阵来旋转图像。

实现步骤

整个程序的逻辑非常清晰:

  1. 加载图像:从文件中读取一张图片。
  2. 创建窗口和滑动条:创建一个窗口来显示图像,并在这个窗口上附加一个范围为 0-360 度的滑动条。
  3. 定义回调函数:创建一个函数,该函数会在滑动条位置改变时被调用。
  4. 执行旋转:在回调函数中,获取滑动条的当前值(角度),计算旋转矩阵,并对原始图像应用旋转变换。
  5. 显示结果:在窗口中更新并显示旋转后的图像。

完整代码示例

下面是实现该功能的完整 C++ 代码。代码结构清晰,并附有详细注释。

cpp 复制代码
#include <iostream>
#include "opencv2/opencv.hpp"

using namespace std;
using namespace cv;

// --- 全局变量 ---
// 窗口名称
const string g_windowName = "图像旋转 (拖动滑动条)"; 
// 原始图像和旋转后的图像
Mat g_srcImage, g_dstImage; 
// 滑动条当前值(角度)和最大值
int g_trackbarValue = 0; 
const int g_trackbarMaxValue = 360; 

/**
 * @brief 滑动条的回调函数
 * @param pos 滑动条的当前位置
 * @param userdata 用户传递的数据(这里未使用)
 */
void onTrackbar(int pos, void* userdata) {
    // 1. 获取旋转中心:图像的几何中心
    Point2f center(g_srcImage.cols / 2.0F, g_srcImage.rows / 2.0F);

    // 2. 计算旋转矩阵
    // 参数:旋转中心、旋转角度 (pos)、缩放因子 (1.0)
    Mat rot = getRotationMatrix2D(center, pos, 1.0);

    // 3. 计算旋转后图像的边界框,以防止图像被裁切
    // 获取旋转后的边界框
    Rect bbox = RotatedRect(center, g_srcImage.size(), pos).boundingRect();
    // 调整旋转矩阵,加入平移变换,使得旋转后的图像能完整显示
    rot.at<double>(0, 2) += bbox.width / 2.0 - center.x;
    rot.at<double>(1, 2) += bbox.height / 2.0 - center.y;

    // 4. 应用仿射变换(旋转)
    // 参数:输入图像、输出图像、变换矩阵、输出图像尺寸
    warpAffine(g_srcImage, g_dstImage, rot, bbox.size());

    // 5. 显示旋转后的图像
    imshow(g_windowName, g_dstImage);
}

int main(int argc, char** argv) {
    // 检查输入
    if (argc != 2) {
        cout << "用法: ./rotate_image <图片路径>" << endl;
        return -1;
    }

    // 1. 加载图像
    g_srcImage = imread(argv[1]);
    if (g_srcImage.empty()) {
        cout << "无法加载图像: " << argv[1] << endl;
        return -1;
    }

    // 2. 创建窗口
    namedWindow(g_windowName, WINDOW_AUTOSIZE);

    // 3. 创建滑动条
    // 参数:滑动条名称、窗口名称、绑定变量指针、最大值、回调函数
    createTrackbar("角度", g_windowName, &g_trackbarValue, g_trackbarMaxValue, onTrackbar);

    // 4. 首次调用回调函数以显示初始图像(0度)
    onTrackbar(0, 0);

    // 5. 等待用户按键后退出
    cout << "拖动滑动条来旋转图像。按任意键退出..." << endl;
    waitKey(0);

    return 0;
}

代码解析
  • 全局变量 : 我们将需要被回调函数 onTrackbar 访问的变量(如源图像 g_srcImage)定义为全局变量,这是一种在回调场景中共享状态的简单方法。
  • 防止裁切 : 如果直接使用 warpAffine,旋转后的图像可能会超出原始图像的边界而被裁切。为了解决这个问题,我们首先计算旋转后图像的边界框(Bounding Box),然后调整旋转矩阵,增加一个平移量,确保整个旋转后的图像都能在新尺寸的画布中完整显示出来。
  • 回调机制 : createTrackbar 是核心。它将一个滑动条与一个整数变量 g_trackbarValue 和一个回调函数 onTrackbar 关联起来。每当用户拖动滑动条时,OpenCV 会自动更新 g_trackbarValue 的值,并用新值作为参数调用 onTrackbar 函数,从而触发图像的重新计算和显示。
如何编译和运行
  1. 保存代码 :将以上代码保存为 rotate_image.cpp

  2. 编译:打开终端,使用 g++ 编译器(需要已配置好 OpenCV 环境)进行编译。

    bash 复制代码
    g++ rotate_image.cpp -o rotate_image `pkg-config --cflags --libs opencv4`

    注意:如果您的 OpenCV 版本是 3.x,请将 opencv4 替换为 opencv

  3. 运行 :准备一张图片(例如 test.jpg),然后执行程序。

    bash 复制代码
    ./rotate_image test.jpg

程序运行后,你会看到一个窗口,窗口上方有一个滑动条。拖动它,图片就会以其中心为轴进行实时旋转!

相关推荐
tan180°2 小时前
MySQL表的操作(3)
linux·数据库·c++·vscode·后端·mysql
学不动CV了2 小时前
ARM单片机启动流程(二)(详细解析)
c语言·arm开发·stm32·单片机·51单片机
彭祥.3 小时前
Jetson边缘计算主板:Ubuntu 环境配置 CUDA 与 cudNN 推理环境 + OpenCV 与 C++ 进行目标分类
c++·opencv·分类
lzb_kkk3 小时前
【C++】C++四种类型转换操作符详解
开发语言·c++·windows·1024程序员节
猫猫的小茶馆4 小时前
【STM32】通用定时器基本原理
c语言·stm32·单片机·嵌入式硬件·mcu·51单片机
Tony沈哲4 小时前
macOS 上为 Compose Desktop 构建跨架构图像处理 dylib:OpenCV + libraw + libheif 实践指南
opencv·算法
胖大和尚5 小时前
clang 编译器怎么查看在编译过程中做了哪些优化
c++·clang
pumpkin845145 小时前
Rust 调用 C 函数的 FFI
c语言·算法·rust
2401_858286116 小时前
125.【C语言】数据结构之归并排序递归解法
c语言·开发语言·数据结构·算法·排序算法·归并排序
钱彬 (Qian Bin)6 小时前
一文掌握Qt Quick数字图像处理项目开发(基于Qt 6.9 C++和QML,代码开源)
c++·开源·qml·qt quick·qt6.9·数字图像处理项目·美观界面