IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 人工智能 -> OpenCV:透视变换/仿射变换 -> 正文阅读

[人工智能]OpenCV:透视变换/仿射变换

//@ 对棋盘格原图透视变换
    //@ 将棋盘格图像变换到屏幕大小,但是周边还是会有边框,图像中的屏幕是没有铺满的,误差
    Mat chessboard_warp;
    cv::warpPerspective(model_var.chessboard_image, chessboard_warp, model_var.warpMatrix, cv::Size(model_var.SCREEN_WIDTH, model_var.SCREEN_HEIGHT), cv::INTER_LINEAR, cv::BORDER_CONSTANT);
    //cv::warpPerspective(model_var.chessboard_image, chessboard_warp, model_var.warpMatrix, cv::Size(5000,5000), cv::INTER_LINEAR, cv::BORDER_CONSTANT);
    cv::namedWindow("chessboard_warp", cv::WINDOW_NORMAL);
    imshow("chessboard_warp", chessboard_warp);


    Mat chessboard_warp_Affine;
    chessboard_warp_Affine.create(model_var.SCREEN_WIDTH, model_var.SCREEN_HEIGHT, CV_16UC1);
    cv::Point2f chess_center;
    chess_center = (model_var.allCorners[17] + model_var.allCorners[24]) / 2;
    cv::Point2i chess_center_warp;
    chess_center_warp = transPoint(chess_center);

    //float offset_x = model_var.SCREEN_WIDTH / 2 - chess_center_warp.x;
    //float offset_y = model_var.SCREEN_HEIGHT / 2 - chess_center_warp.y;
    float offset_x = 800 / 2 - chess_center_warp.x;
    float offset_y = 700 / 2 - chess_center_warp.y;
    Mat trans_mat = (cv::Mat_<float>(2, 3) << 1, 0, offset_x, 0, 1, offset_y);
    cv::warpAffine(chessboard_warp, chessboard_warp_Affine, trans_mat, cv::Size(800,700));
    cv::namedWindow("chessboard_warp_Affine", cv::WINDOW_NORMAL);
    imshow("chessboard_warp_Affine", chessboard_warp_Affine);

给定一个棋盘格图像,灰白色矩形大小为1000*2000,棋盘格在其中心,大小为700*800,每个棋盘格大小为100*100

?通过SB函数可以检测出所有角点,SB函数检测出的角点顺序是颠倒的,需要注意一下,然后就可以将角点坐标与角点在1000*2000中的实际坐标一一对应(这个实际坐标可以方便的推算出来),计算变换矩阵

void getWarpMatrix()
{
    vector<Point2f> dst_coners;
    int ini_x, ini_y;
    ini_x = model_var.CHESS_SIZE_W + (model_var.SCREEN_WIDTH - model_var.CHESS_SIZE_W * (model_var.chessboard_col + 1)) / 2;
    ini_y = model_var.CHESS_SIZE_H + (model_var.SCREEN_HEIGHT - model_var.CHESS_SIZE_H * (model_var.chessboard_row + 1)) / 2;
    
    Point2f tmp;
    for (int i = 0; i < model_var.chessboard_row; i++)
    {
        for (int j = 0; j < model_var.chessboard_col; j++)
        {
            tmp.x = ini_x + j * model_var.CHESS_SIZE_W;
            tmp.y = ini_y + i * model_var.CHESS_SIZE_H;
            dst_coners.push_back(tmp);

#ifdef DEBUG_TEST_IMSHOW_AND_PRINT
            cout << "[" << i << ", " << j << "] ==> (" << tmp.x << ", " << tmp.y << ")" << endl;
#endif // DEBUG_TEST_IMSHOW_AND_PRINT

        }
    }
    model_var.warpMatrix = cv::findHomography(model_var.allCorners, dst_coners);
}

将角点在图像上画出来

cv::drawChessboardCorners(drawChessboardImage, boardSize,
        imageCorners, found); // 找到的角点
cv::namedWindow("drawChessboardImage", cv::WINDOW_NORMAL);
cv::imshow("drawChessboardImage", drawChessboardImage);


进行透视变换,

//@ 对棋盘格原图透视变换
    //@ 将棋盘格图像变换到屏幕大小,但是周边还是会有边框,图像中的屏幕是没有铺满的,误差
    Mat chessboard_warp;
    cv::warpPerspective(model_var.chessboard_image, chessboard_warp, model_var.warpMatrix, cv::Size(model_var.SCREEN_WIDTH, model_var.SCREEN_HEIGHT), cv::INTER_LINEAR, cv::BORDER_CONSTANT);
    //cv::warpPerspective(model_var.chessboard_image, chessboard_warp, model_var.warpMatrix, cv::Size(5000,5000), cv::INTER_LINEAR, cv::BORDER_CONSTANT);
    cv::namedWindow("chessboard_warp", cv::WINDOW_NORMAL);
    imshow("chessboard_warp", chessboard_warp);

由于变换矩阵确定了,变换矩阵是由角点之间相互对应确定的,所以变换前后的原点也是对应的,上图中的红框,所以,不论warpPerspective函数中的那个Size参数如何变化,原点(0,0)不变。Size设为1000*2000的时候,刚好是棋盘格整图,假如Size设小点,则不会显示整个棋盘格图像

?

仿射变换,假如想在透视变换的基础上,截取棋盘格700*800的部分

??

Mat chessboard_warp_Affine;
    chessboard_warp_Affine.create(model_var.SCREEN_WIDTH, model_var.SCREEN_HEIGHT, CV_16UC1);
    cv::Point2f chess_center;
    chess_center = (model_var.allCorners[17] + model_var.allCorners[24]) / 2;
    cv::Point2i chess_center_warp;
    chess_center_warp = transPoint(chess_center);

    //float offset_x = model_var.SCREEN_WIDTH / 2 - chess_center_warp.x;
    //float offset_y = model_var.SCREEN_HEIGHT / 2 - chess_center_warp.y;
    float offset_x = 800 / 2 - chess_center_warp.x;
    float offset_y = 700 / 2 - chess_center_warp.y;
    Mat trans_mat = (cv::Mat_<float>(2, 3) << 1, 0, offset_x, 0, 1, offset_y);
    cv::warpAffine(chessboard_warp, chessboard_warp_Affine, trans_mat, cv::Size(800,700));
    cv::namedWindow("chessboard_warp_Affine", cv::WINDOW_NORMAL);
    imshow("chessboard_warp_Affine", chessboard_warp_Affine);

?

?

  人工智能 最新文章
2022吴恩达机器学习课程——第二课(神经网
第十五章 规则学习
FixMatch: Simplifying Semi-Supervised Le
数据挖掘Java——Kmeans算法的实现
大脑皮层的分割方法
【翻译】GPT-3是如何工作的
论文笔记:TEACHTEXT: CrossModal Generaliz
python从零学(六)
详解Python 3.x 导入(import)
【答读者问27】backtrader不支持最新版本的
上一篇文章      下一篇文章      查看所有文章
加:2022-04-26 11:41:49  更:2022-04-26 11:43:28 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/6 22:58:32-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码