一种简单的图形旋转算法

article/2025/10/6 23:52:11

图形旋转好玩又有实用性, 这里介绍一种简单的图形旋转算法.

具体步骤如下:

1. 首先将原图和旋转图的坐标原点都变换到图形的中心位置处.
2. 历遍旋转图形中的每一个pixel, 将pixel的坐标(j,i)反向旋转映射到原图, 得到原图对应的坐标值(Xr,Yr).
3. 考虑到旋转图的尺寸可能大于原图,这时需要检测(Xr,Yr)是否在原图范围内,如果不是,则忽略下面步骤.
4. (Xr,Yr)通常并不是正好对应到原图中一个整点的像素, 而是会夹在四个像素中(如图), 对四个像素的位置进行插值计算(也即所谓的双线性插值), 可以得到对应位置的RGBA值.

5. 由于内存中的图形数据是以左上角为坐标原点逐行记录的, 我们需将(j,i)变换到此坐标系统下(步骤1的反向操作), 得到其对应的内存位置, 然后将RGBA数值存入此处.
6. 通过逐点计算,最终可以得到完整的旋转后的图形.
7. 待改进处: 这里由于对图形边缘点的采样不够密集,因此得到的旋转图形边缘会出现锯齿现象. 可以细化映射颗粒度,如以1/4像素为单位进行映射计算.

更普遍的情况是,我们需要对原图中的指定区域进行旋转操作, 即进行旋转抠图. 下面是这一算法的C语言实现:

/*-----------------------------------------------------------------
@eimg:   		输入图像
@oimg:   		输出图像(NULL忽略)
@height,width:		输出图像的尺寸
@px,py:		抠图相对原图的位置
@angle:		旋转角度
-------------------------------------------------------------------*/
EGI_IMGBUF* egi_imgbuf_rotBlockCopy2( EGI_IMGBUF *eimg, EGI_IMGBUF *oimg, int height, int width,int px, int py, float angle)
{int i,j;float xr,yr;           int index_out;EGI_IMGBUF      *outimg=NULL;float sina=sin(MATH_PI*angle/180);float cosa=cos(MATH_PI*angle/180);/* 1. Check input eimg */if(eimg==NULL || eimg->imgbuf==NULL || eimg->height<=0 || eimg->width<=0 ) {egi_dpstd("Input holding eimg is NULL or uninitiliazed!\n");return NULL;}/* 2. Input oimg is NULL *//* 注:这里将输出图像的长宽均设为奇数,这样中心点是个整点位置,便于计算. */if(oimg==NULL) {/* Make H/W an odd value, then it has a symmetrical center point. */height |= 0x1;width |= 0x1;if(height<3)height=3;if(width<3)width=3;/* Create an imgbuf accordingly */outimg=egi_imgbuf_create(height, width, 0, 0); /* H, W, alpah, color */if(outimg==NULL) {egi_dpstd("Fail to create outimg!\n");return NULL;}/* Check ALPHA data */if(eimg->alpha==NULL) {free(outimg->alpha);outimg->alpha=NULL;}}/* Input oimg is NOT NULL */else {height=oimg->height;width=oimg->width;outimg=oimg;}/* 3. Clear outimg first. ALPHA to be 0, OR same as eimg. 如果是RGBA格式,可以在这里设置背景色. */egi_imgbuf_resetColorAlpha(outimg, WEGI_COLOR_GRAY2, outimg->alpha==NULL ? -1:0 );  /* img, color, alpha *//* 4. Map back point coordinates to eimg 以旋转图片中心为原点, 历遍旋转图形中的每一个像素点. 反向映射并插值计算得RGBA. */int m=height>>1;int n=width>>1;for(i=-m; i<=m; i++) {for(j=-n; j<=n; j++) {/* 4.1 Map to original coordiante (xr,yr), Origin at center.* 2D point rotation formula ( a positive: Right_hand Rule. ):*      x'=x*cos(a)-y*sin(a)*      y'=x*sin(a)+y*cos(a)*   Coord axis anti_clockwise, point colokwise rotate.*   points coordinates relative to up_right block coord.*  旋转变换公式, 可以查数学手册.*/xr=cosa*j-sina*i;yr=sina*j+cosa*i;/* 4.2 Shift Origin to left_top, as of eimg->imgbuf 变换到原图坐标下 */xr += px;yr += py;/* 4.3 Copy pixel alpha and color.  Limit xr,yr, ignore if they are out of original imgbuf area. */if( xr >= 0.0 && xr <= eimg->width-1 && yr >=0.0 && yr <= eimg->height-1) {index_out=width*(i+m)+(j+n);/* 4.3.1 的到相互邻的4个像数 */int indx1 = eimg->width*floorf(yr)+floorf(xr);int indx2 = eimg->width*floorf(yr)+ceilf(xr);int indx3 = eimg->width*ceilf(yr)+floorf(xr);int indx4 = eimg->width*ceilf(yr)+ceilf(xr);EGI_16BIT_COLOR  pcolor;EGI_8BIT_ALPHA  palpha;float pft;/* 4.3.2 Interpolate within 4 pixles  4点插值得到RGBA */if(eimg->alpha!=NULL && outimg->alpha!=NULL) {egi_16bitColor_interplt4p(eimg->imgbuf[indx1], eimg->imgbuf[indx2],  /* color1, color2 */eimg->imgbuf[indx3], eimg->imgbuf[indx4],       /* color3, color4 */eimg->alpha[indx1], eimg->alpha[indx2],         /* alpha1, alpha2 */eimg->alpha[indx3], eimg->alpha[indx4],         /* alpha3, alpah4 */modff(xr,&pft)*(1<<15), modff(yr, &pft)*(1<<15),  /* f15_x, f15_y */&pcolor, &palpha );outimg->imgbuf[index_out]=pcolor;outimg->alpha[index_out]=palpha;}else {egi_16bitColor_interplt4p(eimg->imgbuf[indx1], eimg->imgbuf[indx2],  /* color1, color2 */eimg->imgbuf[indx3], eimg->imgbuf[indx4],       /* color3, color4 */0, 0, 0, 0,     /* alpha1, alpha2, alpha3, alpha4 */modff(xr,&pft)*(1<<15), modff(yr, &pft)*(1<<15),  /* f15_x, f15_y */&pcolor, NULL );outimg->imgbuf[index_out]=pcolor;}}}}return outimg;
}

(更多代码见 https://github.com/widora/wegi)

效果:


http://chatgpt.dhexx.cn/article/ArZPA01f.shtml

相关文章

多目标跟踪之数据关联算法——匈牙利算法

零、Track和Detection的cost matrix,distance metric。距离计算的方式有如下几种: 距离cost distance metric,track和detection的距离矩阵。 外观距离appearance distance,来自检测切片ROI的网络特征提取;——余弦距离 运动模型距离 马氏距离,来自检测-跟踪的kalman校正…

EAST算法简单解析

前言 最近写了很多算法代码的解析&#xff0c;但是却很少写原理的解析&#xff0c;这段时间学得快忘得也快&#xff0c;所以寻思这几天写几篇学过算法的原理&#xff0c;可能不是很详细但是一定很简单&#xff0c;利于理解。 算法介绍 EAST: An Efficient and Accurate Scen…

定位算法初探

定位算法初探 一、指纹定位算法介绍 指纹定位(finger-printing localization)算法&#xff0c;是基于室内环境复杂&#xff0c;信号反射折射所形成的在不同位置形成的不同的信号强度信息而提出的一套算法。 指纹算法能很好的利用了反射折射所形成的信号信息&#xff0c;离线首…

使用python模拟实现PID控制算法

使用python模拟实现PID控制算法 PID控制算法是工业应用中最广泛算法之一&#xff0c;在闭环系统的控制中&#xff0c;可自动对控制系统进行准确且迅速的校正。 P、I、D分别是“比例&#xff08;proportional&#xff09;、积分&#xff08;integral&#xff09;、微分&#xff…

TCP Nagle算法简述

TCP/IP协议中&#xff0c;无论发送多少数据&#xff0c;总是要在数据前面加上协议头&#xff0c;同时&#xff0c;对方接收到数据&#xff0c;也需要发送ACK表示确认。为了尽可能的利用网络带宽&#xff0c;TCP总是希望尽可能的发送足够大的数据。 &#xff08;一个连接会设置M…

倒角算法推导

推导原理基本很简单&#xff1a; 已知AB&#xff0c; BC两条线段&#xff0c;且交于B点&#xff0c;求倒角半径为 L&#xff0c;AB&#xff0c;BC的倒角 以最短边&#xff08;假定为AB&#xff09;长 LAB&#xff0c; 在BC中&#xff0c;以B为起点&#xff0c;找出与LAB同长度…

[控制算法]

[常用控制算法] 0.博览众长 0.1 视频 1. DR_CAN b站 0.2 文章 1.控制算法整理 0.3 传统 VS 现代控制算法 1. 传统 传统控制算法&#xff1a;PID&#xff0c;模糊&#xff0c;神经网络控制算法。 2. 现代 现代控制算法有比例&#xff0c;LQR算法(用于线性系统)&#x…

求树的直径证明

树的直径&#xff08;最长路&#xff09; 的详细证明 主要是利用了反证法&#xff1a; 假设 s-t这条路径为树的直径&#xff0c;或者称为树上的最长路 现有结论&#xff0c;从任意一点u出发搜到的最远的点一定是s、t中的一点&#xff0c;然后在从这个最远点开始搜&#xff0c;就…

树的直径和树的重心

1.树包括有根树和无根树&#xff0c;有根树是有向图的子图&#xff0c;无根树是无向图的子图&#xff0c;都满足边数等于节点数减一。根是入度为零或没有父亲的节点 2.树的直径&#xff1a;树上最长的简单路径&#xff08;不重复经过点的路径&#xff09; 3.求解算法&#xf…

树的直径总结

树的直径 一、定义 在一棵树中&#xff0c;最远的两个子节点之间的距离被称为树的直径&#xff1b; 链接这两个点的路径被称为树的最长链&#xff1b; 有两种求法&#xff0c;时间复杂度均为 O ( n ) O(n) O(n) &#xff1b; 二、树形DP 1. 状态 由于一个点的最长路通过…

基础算法 - 树的直径

题目地址&#xff1a;https://leetcode-cn.com/problems/tree-diameter/ 1245. 树的直径 难度中等48收藏分享切换为英文接收动态反馈 给你这棵「无向树」&#xff0c;请你测算并返回它的「直径」&#xff1a;这棵树上最长简单路径的 边数。 我们用一个由所有「边」组成的数…

树的直径-c++

题目 实验室里原先有一台电脑(编号为1)&#xff0c;最近氪金带师咕咕东又为实验室购置了N-1台电脑&#xff0c;编号为2到N。每台电脑都用网线连接到一台先前安装的电脑上。但是咕咕东担心网速太慢&#xff0c;他希望知道第i台电脑到其他电脑的最大网线长度&#xff0c;但是可怜…

求树的直径算法以及证明

以下为两次dfs&#xff08;bfs&#xff09;的做法以及正确性证明。 算法步骤 &#xff08;1&#xff09;任取树上一点S&#xff0c;以S为源点BFS得S到各个顶点的d值&#xff1b; &#xff08;2&#xff09;取d值最大者之一为P&#xff0c;再以P为源点BFS得P到各个顶点的d值&am…

求树的直径

树的直径&#xff0c;即树上的最长路径&#xff0c;显然&#xff0c;树的直径可以有很多条&#xff08;考虑一棵菊花&#xff09;。 接下来我们考虑如何求出一棵树的直径。有很多种O(n)的算法。 算法1&#xff1a;我们任取树中的一个节点x&#xff0c;找出距离它最远的点y&am…

数据结构 树的直径

学习的最大理由是想摆脱平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;迟一天就多一天平庸的困扰。 学习日记 目录 学习日记 一、定义 二、两次DFS 定理&#xff1a; 反证法证明&#xff1a; 1、若y在d(t,s)上 2、若y不在d(s,t)上&#xff0c;且d(y,z)与d(s.t)…

树的直径(最长的简单路径)

题解&#xff1a;分析一下&#xff0c;由于是树&#xff0c;所以两点之间的路径有且只有一条&#xff0c;为了求出欧拉路&#xff0c;所以必然会向回走&#xff0c;从递归的角度来看&#xff0c;假设x看作一个树根&#xff0c;有t个孩子y1…yt。其中每个孩子为根的子树欧拉路都…

树的直径概念及求解

文章目录 1. 使用两次DFS求得树的直径2. 使用树形DP求得树的直径3. 性质4. 参考文献和习题 树上任意两节点之间最长的简单路径即为树的「直径」。显然&#xff0c;一棵树可以有多条直径&#xff0c;他们的长度相等。可以用两次 D F S / B F S DFS/BFS DFS/BFS 或者树形 D P D…

树的直径两种求法

首先先介绍一下什么是树的直径&#xff0c;树的直径就是树中所有最短路经距离的最大值。 求树的直径通常有两种方法&#xff0c;一种是通过两次搜索&#xff08;bfs和dfs均可&#xff09;&#xff0c;另一种就是通过树形dp来求了。 先来介绍一下用两次深搜来求树的直径&#x…

树的直径的概念

树的直径的定义: 在一棵树中&#xff0c;每一条边都有权值&#xff0c;树中的两个点之间的距离&#xff0c;定义为连接两点的路径上边权之和&#xff0c; 那么树上最远的两个点&#xff0c;他们之间的距离&#xff0c;就被称之为&#xff0c;树的直径。 树的直径的别称&#x…

树的直径

【定义】 我们将一棵树T ( V&#xff0c;E )的直径定义为maxδ ( u&#xff0c;v ) ( u&#xff0c;v ∈ V )&#xff0c;也就是说&#xff0c;树中所有最短路径距离的最大值即为树的直径。 【做法】 例题传送门Cow Marathon&#xff08;POJ 1985&#xff09; 对于树的直径…