美文网首页
图像旋转算法与实现

图像旋转算法与实现

作者: alleNjord | 来源:发表于2018-12-09 22:23 被阅读0次

图像旋转是指图像以某一点为中心旋转一定的角度,形成一幅新的图像的过程。当然这个点通常就是图像的中心。既然是按照中心旋转,自然会有这样一个属性:旋转前和旋转后的点离中心的位置不变.

根据这个属性,我们可以得到旋转后的点的坐标与原坐标的对应关系。由于原图像的坐标是以左上角为原点的,所以我们先把坐标转换为以图像中心为原点。假设原图像的宽为w,高为h,(x0,y0)为原坐标内的一点,转换坐标后的点为(x1,y1)。那么不难得到:

x1 = x0 - w/2; y1 = -y0 + h/2;

在新的坐标系下,假设点(x0,y0)距离原点的距离为r,点与原点之间的连线与x轴的夹角为b,旋转的角度为a,旋转后的点为(x1,y1), 如下图所示。

image

那么有以下结论:

x0=rcosb;y0=rsinb

x1 = rcos(b-a) = rcosbcosa+rsinbsina=x0cosa+y0sina;

y1=rsin(b-a)=rsinbcosa-rcosbsina=-x0sina+y0cosa;

得到了转换后的坐标,我们只需要把这些坐标再转换为原坐标系即可。这里还有一点要注意,旋转后的图像的长和宽会发生变化,因此要计算新图像的长和宽。

#include "stdafx.h"
#include <stdio.h>
#include <string>
#include <math.h>
#include <windows.h>
 using namespace std;

 #define PI 3.1415926535
//角度到弧度转化
#define RADIAN(angle) ((angle)*PI/180.0)

void Rotation(const string& srcFile,const string& desFile,int angle)
{
    BITMAPFILEHEADER bmfHeader;
    BITMAPINFOHEADER bmiHeader;
    
    FILE *pFile;
    if ((pFile = fopen(srcFile.c_str(),"rb")) == NULL)
    {
        printf("open bmp file error.");
        exit(-1);
    }
    //读取文件和Bitmap头信息
    fseek(pFile,0,SEEK_SET);
    fread(&bmfHeader,sizeof(BITMAPFILEHEADER),1,pFile);
    fread(&bmiHeader,sizeof(BITMAPINFOHEADER),1,pFile);
    //先不支持小于16位的位图
    int bitCount = bmiHeader.biBitCount;
    if (bitCount < 16)
    {        
        exit(-1);
    }
    int srcW = bmiHeader.biWidth;
    int srcH = bmiHeader.biHeight;
    //原图像每一行去除偏移量的字节数
    int lineSize = bitCount * srcW / 8;
    //偏移量,windows系统要求每个扫描行按四字节对齐
    int alignBytes = ((bmiHeader.biWidth * bitCount + 31) & ~31) / 8L
        - bmiHeader.biWidth * bitCount / 8L;
    //原图像缓存
    int srcBufSize = lineSize * srcH;
    BYTE* srcBuf = new BYTE[srcBufSize];
    int i,j;
    //读取文件中数据
    for (i = 0; i < srcH; i++)
    {        
        fread(&srcBuf[lineSize * i],lineSize,1,pFile);
        fseek(pFile,alignBytes,SEEK_CUR);
    }
    //以图像中心为原点左上角,右上角,左下角和右下角的坐标,用于计算旋转后的图像的宽和高
    POINT pLT,pRT,pLB,pRB;
    pLT.x = -srcW/2;pLT.y = srcH/2;
    pRT.x = srcW/2;pRT.y = srcH/2;
    pLB.x = -srcW/2;pLB.y = -srcH/2;
    pRB.x = srcW/2; pRB.y = -srcH/2;
    //旋转之后的坐标
    POINT pLTN,pRTN,pLBN,pRBN;
    double sina = sin(RADIAN(angle));
    double cosa = cos(RADIAN(angle));
    pLTN.x = pLT.x*cosa + pLT.y*sina;    
    pLTN.y = -pLT.x*sina + pLT.y*cosa;
    pRTN.x = pRT.x*cosa + pRT.y*sina;
    pRTN.y = -pRT.x*sina + pRT.y*cosa;
    pLBN.x = pLB.x*cosa + pLB.y*sina;
    pLBN.y = -pLB.x*sina + pLB.y*cosa;
    pRBN.x = pRB.x*cosa + pRB.y*sina;
    pRBN.y = -pRB.x*sina + pRB.y*cosa;
    //旋转后图像宽和高
    int desWidth = max(abs(pRBN.x - pLTN.x),abs(pRTN.x - pLBN.x));
    int desHeight = max(abs(pRBN.y - pLTN.y),abs(pRTN.y - pLBN.y));
    //分配旋转后图像的缓存
    int desBufSize = ((desWidth * bitCount + 31) / 32) * 4 * desHeight;
    BYTE *desBuf = new BYTE[desBufSize];
    //将所有像素都预置为白色
    memset(desBuf,255,desBufSize);
    //新图像每一行字节数,带有偏移量
    int desLineSize = ((desWidth * bitCount + 31) / 32) * 4;        
    //通过新图像的坐标,计算对应的原图像的坐标
    for (i = 0; i < desHeight; i++)
    {        
        for (j = 0; j < desWidth; j++)
        {
            //转换到以图像为中心的坐标系,并进行逆旋转
            int tX = (j - desWidth / 2)*cos(RADIAN(360 - angle)) + (-i + desHeight / 2)*sin(RADIAN(360 - angle));
            int tY = -(j - desWidth / 2)*sin(RADIAN(360 - angle)) + (-i + desHeight / 2)*cos(RADIAN(360 - angle));
            //如果这个坐标不在原图像内,则不赋值
            if (tX > srcW / 2 || tX < -srcW / 2 || tY > srcH / 2 || tY < -srcH / 2)
            {
                continue;
            }
            //再转换到原坐标系下
            int tXN = tX + srcW / 2; int tYN = abs(tY - srcH / 2);
            //值拷贝
            memcpy(&desBuf[i * desLineSize + j * bitCount / 8],&srcBuf[tYN * lineSize + tXN * bitCount / 8],3);            
        }
    }

    //创建目标文件
    HFILE hfile = _lcreat(desFile.c_str(),0);    
    //文件头信息
    BITMAPFILEHEADER nbmfHeader;    
    nbmfHeader.bfType = 0x4D42;
    nbmfHeader.bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)
        + desWidth * desHeight * bitCount / 8;
    nbmfHeader.bfReserved1 = 0;
    nbmfHeader.bfReserved2 = 0;
    nbmfHeader.bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER);
    //Bitmap头信息
    BITMAPINFOHEADER   bmi; 
    bmi.biSize=sizeof(BITMAPINFOHEADER); 
    bmi.biWidth=desWidth; 
    bmi.biHeight=desHeight; 
    bmi.biPlanes=1; 
    bmi.biBitCount=bitCount; 
    bmi.biCompression=BI_RGB; 
    bmi.biSizeImage=0; 
    bmi.biXPelsPerMeter=0; 
    bmi.biYPelsPerMeter=0; 
    bmi.biClrUsed=0; 
    bmi.biClrImportant=0; 
    
    //写入文件头信息
    _lwrite(hfile,(LPCSTR)&nbmfHeader,sizeof(BITMAPFILEHEADER));
    //写入Bitmap头信息
    _lwrite(hfile,(LPCSTR)&bmi,sizeof(BITMAPINFOHEADER));
    //写入图像数据
    _lwrite(hfile,(LPCSTR)desBuf,desBufSize);
    _lclose(hfile);
}

int main(int argc, char* argv[])
{
    FILE *pFile;
    if ((pFile = fopen("e://t.bmp","rb")) == NULL)
    {
        printf("open bmp file error.");
        return -1;
    }
    string srcFile("e://t.bmp");
    string desFile("e://Rotation.bmp");
    Rotation(srcFile,desFile,150);
    system("pause");
    return 0;
}

相关文章

  • 图像旋转算法与实现

    图像旋转是指图像以某一点为中心旋转一定的角度,形成一幅新的图像的过程。当然这个点通常就是图像的中心。既然是按照中心...

  • 8 ARTS打卡第八周(2019-09-17)

    Algorithm 本周算法:48.旋转图像(未解出)题解思路:使用两次翻转来实现旋转效果,首先第一次翻转,将对应...

  • JavaScript 算法(旋转图像)

    给定一个 *n *× n 的二维矩阵表示一个图像。 将图像顺时针旋转 90 度。 说明: 你必须在原地旋转图像,这...

  • OpenCV

    原图像保边滤波算法集锦--MeanShift滤波算法与实现https://blog.csdn.net/Trent1...

  • OpenCV实现图像旋转

    不知道是不是因为太简单,OpenCV不屑于去封装还是怎么回事,OpenCV中并没有一个API是专门用于实现图像任意...

  • 初级算法-数组-旋转图像

    给定一个 n × n 的二维矩阵 matrix 表示一个图像。请你将图像顺时针旋转 90 度。 你必须在 原地[h...

  • 编程作业(七)

    K均值算法与主成分分析算法 K均值分析算法 在本部分练习中,你将实现K均值算法并将该算法用于图像压缩。最初,你通过...

  • C语言—实现射线检测多边形碰撞

    以前,使用旋转分离轴实现过, 矩形旋转碰撞,OBB方向包围盒算法实现。但这个算法,本身有点复杂,并且在边越多的时候...

  • 校园招聘2017

    网易招聘 图像算法工程师(北京) 岗位描述:1、负责实现并优化编解码及压缩等图像算法;2、负责图像相关的技术、系统...

  • 图像缩放算法实现与对比

    算法原理 最临近插值算法原理图示: 选取对应原图位置距离最近的点,当然,距离相等的点的选取,需要自定一些规则。模块...

网友评论

      本文标题:图像旋转算法与实现

      本文链接:https://www.haomeiwen.com/subject/xmclhqtx.html