使用c++程序,实现图像平移变换,图像缩放、图像裁剪、图像对角线镜像以及图像的旋转

数字图像处理–实验三A图像的基本变换

实验内容

A实验:
(1)使用VC++设计程序:实现图像平移变换,图像缩放、图像裁剪、图像对角线镜像。
(2)使用VC++设计程序:对一幅高度与宽度均相等的图像,实现逆时针90度旋转。—这个直接使用B实验的代码也可以完成(B–任意大小图片旋转任意角度)

一、图像平移变换

1. 实验原理

图像的平移:通过直角坐标系的平移变换公式:
x’ = x +dx
y’ = y + dy

注:
(x,y)为源图像的坐标,(x’, y’)为新图像的坐标,dx对应x的偏移量,dy对应y的偏移量。即:平移之后新图像上坐标为(x’, y’)的像素点的颜色值,应该等于原图像上坐标为(x, y)的像素点的颜色值。

2. 实验代码

void CImageProcessingView::OnGeoTranslation()
{// 实验 图像平移//MessageBox("请在这里添加图像平移的代码");// 获得当前文档对象CImageProcessingDoc* pDoc = GetDocument();// 判断图像是否已被加载if( pDoc->m_pDibInit->IsEmpty() ){MessageBox("图像未加载");return;}int width = pDoc->m_pDibInit->GetWidth();int height = pDoc->m_pDibInit->GetHeight();int bitCount = pDoc->m_pDibInit->GetBitCount();// 将 m_pDibInit 拷贝至 m_pDibTestpDoc->m_pDibTest->CloneDib(pDoc->m_pDibInit);// 设置 pDoc->m_pDibTest 为全白图像int i, j;RGBQUAD rgbQuad1;rgbQuad1.rgbBlue = 255;rgbQuad1.rgbGreen = 255;rgbQuad1.rgbRed = 255;rgbQuad1.rgbReserved = 0;for(i=0; i<width; i++){for(j=0; j<height; j++){pDoc->m_pDibTest->SetPixelColor(i,j,&rgbQuad1);}}//************************图像平移****************************//RGBQUAD Quad;//新的像素点float  translatepixel = 2 ;int newwidth=width*translatepixel;//移动后的新的画布的大小int newheight=height*translatepixel;pDoc->m_pDibTest->SetWidthHeight(newwidth,newheight);//重设画布大小//将像素点的值进行移动for(i=0;i<width;i++)for(j=0;j<height;j++){Quad =pDoc->m_pDibInit->GetPixelColor(i,j);//获取原图像的像素点的值pDoc->m_pDibTest->SetPixelColor(i+256,j,&Quad);}
//*********************************************************************// // 交换 m_pDibInit 与 m_pDibTest 指针CDib* pTmp = pDoc->m_pDibInit;pDoc->m_pDibInit = pDoc->m_pDibTest;pDoc->m_pDibTest = pTmp;// 设置脏标记pDoc->SetModifiedFlag(TRUE);// 更新视图pDoc->UpdateAllViews(NULL);
}

实现现象

将图片右移了256个像素点
在这里插入图片描述

二、图像缩放

1. 缩放原理

设原图像大小为宽度M、高度N,调整后宽度为k1×M、高度为k2×N,则:
Img.New(x,y) = Img.Old(x/k1, y/k2)
这样就可以实现图像的缩放

2. 实验代码

void CImageProcessingView::OnGeoResizing()
{// 实验 图像缩放//MessageBox("请在这里添加图像缩放的代码");// 获得当前文档对象CImageProcessingDoc* pDoc = GetDocument();// 判断图像是否已被加载if( pDoc->m_pDibInit->IsEmpty() ){MessageBox("图像未加载");return;}int width = pDoc->m_pDibInit->GetWidth();int height = pDoc->m_pDibInit->GetHeight();int bitCount = pDoc->m_pDibInit->GetBitCount();// 将 m_pDibInit 拷贝至 m_pDibTestpDoc->m_pDibTest->CloneDib(pDoc->m_pDibInit);// 考虑将图像放大两倍的情况 float nResizing = 0.6;//大于1则为放大,小于1则为缩小// 获得新的图像高度int newWidth = width*nResizing;int newHeight = height*nResizing;pDoc->m_pDibTest->SetWidthHeight(newWidth, newHeight);//***************************图像的放大与缩小***********////间隔采样int i=0;int j=0;RGBQUAD Quad1;for(i=0;i<newWidth;i++)for(j=0;j<newHeight;j++){Quad1=pDoc->m_pDibInit->GetPixelColor(i/nResizing,j/nResizing);pDoc->m_pDibTest->SetPixelColor(i,j,&Quad1);}// 交换 m_pDibInit 与 m_pDibTest 指针CDib* pTmp = pDoc->m_pDibInit;pDoc->m_pDibInit = pDoc->m_pDibTest;pDoc->m_pDibTest = pTmp; // 设置脏标记pDoc->SetModifiedFlag(TRUE);// 更新视图pDoc->UpdateAllViews(NULL);
}

实验现象

将图片变为原来的0.6倍
在这里插入图片描述

三、图像裁剪

裁剪原理

图像裁剪是从原始图像中选择或提取感兴趣的区域,以产生新的图像。这个过程可以通过调整图像的像素坐标来实现。以下是图像裁剪的基本原理:

  1. 选择裁剪区域: 首先,确定你希望保留的图像区域。这可以通过指定裁剪区域的坐标、宽度和高度来完成。裁剪区域通常以左上角和右下角的坐标、宽度和高度来定义。

  2. 调整坐标: 对于裁剪区域内的每个像素,调整其坐标以匹配裁剪后的图像。如果裁剪区域的左上角坐标是 ( x start , y start ) (x_{\text{start}}, y_{\text{start}}) (xstart,ystart),那么裁剪后的图像中的像素 ( i , j ) (i, j) (i,j)的新坐标是 ( i − x start , j − y start ) (i-x_{\text{start}}, j - y_{\text{start}}) (ixstart,jystart)

  3. 创建新图像: 使用裁剪后的坐标,从原始图像中提取像素值,并将它们组合成新的图像。新图像的宽度和高度将是裁剪区域的宽度和高度。

实验代码c++

void CImageProcessingView::OnGeoCut()
{// 实验 图像裁剪// 获得当前文档对象CImageProcessingDoc* pDoc = GetDocument();// 判断图像是否已被加载if( pDoc->m_pDibInit->IsEmpty() ){MessageBox("图像未加载");return;}// 将 m_pDibInit 拷贝至 m_pDibTestpDoc->m_pDibTest->CloneDib(pDoc->m_pDibInit);// 设置裁剪的起始坐标点int x1 = pDoc->m_pDibInit->m_lpBMIH->biWidth/4;int y1 = pDoc->m_pDibInit->m_lpBMIH->biHeight/3;// 设置裁剪的宽度与高度int cx = pDoc->m_pDibInit->m_lpBMIH->biWidth/2;int cy = pDoc->m_pDibInit->m_lpBMIH->biHeight/3;// 将 m_pDibTest 的宽度与高度 按照裁剪的要求进行设置pDoc->m_pDibTest->SetWidthHeight(cx, cy);// 复制像素点int i,j;RGBQUAD rgbQuad;for(i=0; i<cx; i++)for(j=0; j<cy; j++){rgbQuad = pDoc->m_pDibInit->GetPixelColor(i+x1, j+y1);pDoc->m_pDibTest->SetPixelColor(i, j, &rgbQuad);}// 交换 m_pDibInit 与 m_pDibTest 指针CDib* pTmp = pDoc->m_pDibInit;pDoc->m_pDibInit = pDoc->m_pDibTest;pDoc->m_pDibTest = pTmp; // 设置脏标记pDoc->SetModifiedFlag(TRUE);// 更新视图pDoc->UpdateAllViews(NULL);
}

实验现象

裁剪出中间的一部分:
在这里插入图片描述

四、图像对角线镜像

对角线原理

图像的对角镜像是指通过图像的对角线进行翻转,从而得到镜像效果。具体来说,对角镜像的原理是将图像中的每个像素与其对角位置上的像素进行交换。

对于一个二维图像,坐标为 ( i , j ) (i, j) (i,j) 的像素与坐标为 ( j , i ) (j, i) (j,i) 的像素进行交换。这个操作实际上是关于对角线翻转。如果图像的宽度和高度相等,对角镜像就是将图像沿着其主对角线翻转。

以下是一个简单的示例说明对角镜像的原理:

假设有一个3x3的图像:

1 2 3
4 5 6
7 8 9

对角线上的元素是 (1, 5, 9),将它们与对应的对角位置上的元素进行交换:

1 4 7
2 5 8
3 6 9

这就是对角线翻转后的图像。对于更大的图像,同样的原理适用。

在计算机图像处理中,对角线镜像通常涉及到图像的像素交换或者矩阵操作。这可以通过遍历图像的每个像素并进行交换操作来实现。

实验代码

void CImageProcessingView::OnDiagonalMirror()
{ // 实验 对角线镜像显示图像//MessageBox("请在这里添加图像对角线镜像的代码");// 可以参考:CImageProcessingView::OnGeoVerticalMirror()// 获得当前文档对象CImageProcessingDoc* pDoc = GetDocument();// 判断图像是否已被加载if( pDoc->m_pDibInit->IsEmpty() ){MessageBox("图像未加载");return;}int width = pDoc->m_pDibInit->GetWidth();int height = pDoc->m_pDibInit->GetHeight();int bitCount = pDoc->m_pDibInit->GetBitCount();if( width!=height ){MessageBox("图像长度与宽度不一致");return;}// 将 m_pDibInit 拷贝至 m_pDibTestpDoc->m_pDibTest->CloneDib(pDoc->m_pDibInit);
//*************************图像的对角镜像******************//int i,j;RGBQUAD quad;for(i=0;i<width;i++)for(j=0;j<height;j++){quad = pDoc->m_pDibInit->GetPixelColor(i, j);pDoc->m_pDibTest->SetPixelColor(j, i, &quad);  //将横竖坐标对换即可}// 交换 m_pDibInit 与 m_pDibTest 指针CDib* pTmp = pDoc->m_pDibInit;pDoc->m_pDibInit = pDoc->m_pDibTest;pDoc->m_pDibTest = pTmp; // 设置脏标记pDoc->SetModifiedFlag(TRUE);// 更新视图pDoc->UpdateAllViews(NULL);
}

实验现象

在这里插入图片描述

五、 任意大小图片旋转任意角度

原先的实验要求是实现宽和高相等的图片的旋转,但是后来B实验的要求是任意角度,就改写了代码,升级了一下

旋转原理

图像的旋转是通过应用旋转矩阵来实现的。旋转矩阵对应于二维平面上的坐标变换,其中旋转角度 (\theta) 决定了变换的角度。对于给定的坐标 ((X, Y)),应用旋转矩阵可以得到旋转后的坐标 ((X’, Y’))。

旋转矩阵如下所示:

[ X ′ Y ′ ] = [ cos ⁡ ( θ ) − sin ⁡ ( θ ) sin ⁡ ( θ ) cos ⁡ ( θ ) ] [ X Y ] \begin{bmatrix}X' \\Y'\end{bmatrix}=\begin{bmatrix}\cos(\theta) & -\sin(\theta) \\\sin(\theta) & \cos(\theta)\end{bmatrix}\begin{bmatrix}X \\Y\end{bmatrix} [XY]=[cos(θ)sin(θ)sin(θ)cos(θ)][XY]

对应的坐标变换公式是:

X ′ = X ⋅ cos ⁡ ( θ ) − Y ⋅ sin ⁡ ( θ ) X' = X \cdot \cos(\theta) - Y \cdot \sin(\theta) X=Xcos(θ)Ysin(θ)

Y ′ = X ⋅ sin ⁡ ( θ ) + Y ⋅ cos ⁡ ( θ ) Y' = X \cdot \sin(\theta) + Y \cdot \cos(\theta) Y=Xsin(θ)+Ycos(θ)

这个公式描述了绕原点进行旋转的情况。如果要围绕其他点旋转,则需要先将坐标平移到该点,执行旋转,然后再将坐标平移到原来的位置。

在计算机图像处理中,旋转通常涉及对图像的每个像素应用坐标变换,以实现整体图像的旋转效果。

实验代码

void CImageProcessingView::OnGeoRotation()
{// 实验 图像旋转//MessageBox("请在这里添加图像旋转的代码");// 获得当前文档对象CImageProcessingDoc* pDoc = GetDocument();// 判断图像是否已被加载if( pDoc->m_pDibInit->IsEmpty() ){MessageBox("图像未加载");return;}int width = pDoc->m_pDibInit->GetWidth();int height = pDoc->m_pDibInit->GetHeight();int bitCount = pDoc->m_pDibInit->GetBitCount();pDoc->m_pDibTest->CloneDib(pDoc->m_pDibInit);int newwidth = width * (sqrt(2) + 1);int newheight = height * (sqrt(2) + 1);pDoc->m_pDibTest->SetWidthHeight(newwidth,newheight); //重设大小为对角线长度double angle = M_PI / 2; //任意角度RGBQUAD quad;// 设置 pDoc->m_pDibTest 为全白图像int i, j;RGBQUAD rgbQuad1;rgbQuad1.rgbBlue = 255;rgbQuad1.rgbGreen = 255;rgbQuad1.rgbRed = 255;rgbQuad1.rgbReserved = 0;for (i = 0; i < newwidth; i++){for (j = 0; j < newheight; j++){pDoc->m_pDibTest->SetPixelColor(i, j, &rgbQuad1);}}for ( i = 0; i < width; i++)for ( j = 0; j < height; j++){quad = pDoc->m_pDibInit->GetPixelColor(i, j);int x = 0;x = -(i * cos(angle) + j * sin(angle));int y = 0;y = (i * sin(angle) + j * cos(angle));pDoc->m_pDibTest->SetPixelColor(x, y, &quad);} // 交换 m_pDibInit 与 m_pDibTest 指针CDib* pTmp = pDoc->m_pDibInit;pDoc->m_pDibInit = pDoc->m_pDibTest;pDoc->m_pDibTest = pTmp; // 设置脏标记pDoc->SetModifiedFlag(TRUE);// 更新视图pDoc->UpdateAllViews(NULL);
}

实验现象

逆时针旋转90°
在这里插入图片描述

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

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

相关文章

linux 系统下文本编辑常用的命令

一、是什么 Vim是从 vi 发展出来的一个文本编辑器&#xff0c;代码补全、编译及错误跳转等方便编程的功能特别丰富&#xff0c;在程序员中被广泛使用。 简单的来说&#xff0c; vi 是老式的字处理器&#xff0c;不过功能已经很齐全了&#xff0c;但是还是有可以进步的地方 而…

Playwright UI 自动化测试实战

&#x1f4e2;专注于分享软件测试干货内容&#xff0c;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指正&#xff01;&#x1f4e2;交流讨论&#xff1a;欢迎加入我们一起学习&#xff01;&#x1f4e2;资源分享&#xff1a;耗时200小时精选的「软件测试」资…

Kylin-Server-V10-SP3+Gbase+宝兰德信创环境搭建

目录 一、Kylin-Server-V10-SP3 安装1.官网下载安装包2.创建 VMware ESXi 虚拟机3.加载镜像&#xff0c;安装系统 二、Gbase 安装1.下载 Gbase 安装包2.创建组和用户、设置密码3.创建目录4.解压包5.安装6.创建实例7.登录8.常见问题 三、宝兰德安装1.获取安装包2.解压安装3.启动…

参考意义大。4+巨噬细胞相关生信思路,简单易复现。

今天给同学们分享一篇生信文章“Angiogenesis regulators S100A4, SPARC and SPP1 correlate with macrophage infiltration and are prognostic biomarkers in colon and rectal cancers”&#xff0c;这篇文章发表在Front Oncol期刊上&#xff0c;影响因子为4.7。 结果解读&a…

Java Stream 的常用API

Java Stream 的常用API 遍历&#xff08;forEach&#xff09; package com.liudashuai;import java.util.ArrayList; import java.util.List;public class Test {public static void main(String[] args) {List<Person> userList new ArrayList<>();userList.ad…

使用Filebeat+Kafka+Logstash+Elasticsearch构建日志分析系统

随着时间的积累&#xff0c;日志数据会越来越多&#xff0c;当您需要查看并分析庞杂的日志数据时&#xff0c;可通过FilebeatKafkaLogstashElasticsearch采集日志数据到Elasticsearch中&#xff0c;并通过Kibana进行可视化展示与分析。本文介绍具体的实现方法。 一、背景信息 …

C语言不可不敲系列:跳水比赛排名问题

目录 1题干&#xff1a; 2解题思路&#xff1a; 3代码: 4运行结果: 5总结: 1题干&#xff1a; 5位运动员参加了10米台跳水比赛&#xff0c;有人让他们预测比赛结果 A选手说&#xff1a;B第二&#xff0c;我第三&#xff1b; B选手说&#xff1a;我第二&#xff0c;E第四&am…

【LeetCode刷题-滑动窗口】--1456.定长子串中元音的最大数目

1456.定长子串中元音的最大数目 方法&#xff1a;使用滑动窗口 class Solution {public int maxVowels(String s, int k) {int n s.length();int sum 0;for(int i 0;i<k;i){sum isVowel(s.charAt(i));}int ans sum;for(int i k;i<n;i){sum sum isVowel(s.charAt…

MHA实验和架构

什么是MHA&#xff1f; masterhight availabulity&#xff1a;基于主库的高可用环境下可以实现主从复制、故障切换 MHA的主从架构最少要一主两从 MHA的出现是为了解决MySQL的单点故障问题。一旦主库崩溃&#xff0c;MHA可以在0-30秒内自动完成故障切换。 MHA的数据流向和工…

钡铼技术4G RTU采集器在智慧农业灌溉控制中的使用介绍

随着科技的不断发展&#xff0c;智慧农业已经成为现代农业发展的重要方向之一。在智慧农业中&#xff0c;灌溉控制是至关重要的环节&#xff0c;而4G RTU采集器作为一种先进的数据采集设备&#xff0c;为智慧农业灌溉控制提供了全新的解决方案。本文将就钡铼技术有限公司的4G R…

算法--搜索与图

这里写目录标题 主要内容DFS思想 BFS思想 DFS与BFS的比较一级目录二级目录二级目录二级目录 一级目录二级目录二级目录二级目录 一级目录二级目录二级目录二级目录 主要内容 DFS 思想 会优先向深处搜索 一旦到达最深处 那么会回溯 但是在回溯的过程中 会边回溯边观察是否有能继…

武汉凯迪正大—锂电池均衡维护仪

产品概况 KDZD885C 电池容量平衡测试系统&#xff0c;主要用于锂电池箱充放电测试及均衡维护&#xff0c;解决锂电池包单芯电压不均衡的痛点&#xff0c;用于快速解决锂电池电压不一致的难题,适用于各锂电池模组电压等级&#xff0c;集单芯放电&#xff0c;充电&#xff0c;均…

linux openlab搭建web网站

网站需求&#xff1a; 1.基于域名 www.openlab.com 可以访问网站内容为 welcome to openlab!!! 2.给该公司创建三个子界面分别显示学生信息&#xff0c;教学资料和缴费网站&#xff0c; 1、基于 www.openlab.com/student 网站访问学生信息&#xff0c; 2、基于 www.openlab…

Chrome版本对应Selenium版本

1.获得浏览器版本号和驱动 浏览器版本: 119.0.6045.124 浏览器驱动版本: 119.0.6043.1 / 120.0.6051.0 访问 https://vikyd.github.io/download-chromium-history-version/ 2. 安装selenium pip install selenium4.1.1 -i http://pypi.mirrors.ustc.edu.cn/simple/ --trusted…

DAY53 1143.最长公共子序列 + 1035.不相交的线 + 53. 最大子序和

1143.最长公共子序列 题目要求&#xff1a;给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长公共子序列的长度。 一个字符串的 子序列 是指这样一个新的字符串&#xff1a;它是由原字符串在不改变字符的相对顺序的情况下删除某些字符&#xff08;也可以不删…

记录:unity脚本的编写6.0

目录 unity UI系统添加ui编写脚本 unity UI系统 在日常的游戏或者别的什么活动中&#xff0c;ui总是必不可少的一项&#xff0c;在java中也有关于GUI的内容&#xff0c;unity也不例外&#xff0c;这次就使用脚本控制在unity添加的各种ui组件&#xff0c;使他们可以完成一些我们…

MTK手机平台充电原理

EPT GPIO初始化文件 bsp_gpio_ept_config.c 1 知识点总结 1.1 Official 参考充电电路 Figure 1-1 参考电路 VCHG&#xff1a;USB正极 VCDT&#xff1a;VCHG Charger Detect充电电压检测脚 ISENSE&#xff1a;充电电流检测电阻的正极 BATSNS&#xff1a;充电电流检测电阻的负极 …

桌面云架构讲解(VDI、IDV、VOI/TCI、RDS)

目录 云桌面架构 VDI 虚拟桌面基础架构 IDV 智能桌面虚拟化 VOI/TCI VOI 虚拟系统架构 TCI 透明计算机架构 RDS 远程桌面服务 不同厂商云桌面架构 桌面传输协议 什么是云桌面 桌面云是虚拟化技术成熟后发展起来的一种应用&#xff0c;桌面云通常也称为云桌面、VDI等 …

Selenium+JQuery定位方法及应用

SeleniumJQuery定位方法及应用 1 JQuery定位说明1.1 JQuery定位方法1.2 JQuery最常用的三个操作1.3 JQuery一个示例1.3.1 用户名输入框1.3.2 密码输入框1.3.3 登陆按钮1.3.4 完整代码 2 JQuery选择器2.1 常用选择器列表2.2 思考 1、关于Selenium提供了很多元素定位方法&#xf…