图像编程学习笔记5——图像镜像

以下文字内容copy于<<数字图像处理编程入门>>,code为自己实现,是win32控制台程序。

 

镜象(mirror)分水平镜象和垂直镜象两种。图2.2的水平镜象和垂直镜象分别如图2.13和图2.14所示

图像编程学习笔记5——图像镜像

图2.13   图2.2的水平镜象

                                                                                 图像编程学习笔记5——图像镜像

 

图2.14   图2.2的垂直镜象

镜象的变换矩阵很简单。设原图宽为w,高为h,变换后,图的宽和高不变。

水平镜象的变化矩阵为:

                                                                                         图像编程学习笔记5——图像镜像 

 

(2.10)

垂直镜象的变化矩阵为:

                                                                                         图像编程学习笔记5——图像镜像 

 

(2.11)

镜象变换的源代码如下,因为和平移的那段程序很类似,程序中的注释就简单一些。

[cpp] view plaincopy
 
    1. /** 
    2. * 程序名: Mirror.cpp 
    3. * 功  能: 实现灰度图像的水平镜像和垂直镜像 
    4. *         测试图片test.bmp放在工程目录下 
    5. */  
    6. #include <iostream>  
    7. #include <windows.h>  
    8. #include <fstream>  
    9. #include <cstring>  
    10. using namespace std;  
    11. BITMAPFILEHEADER bmpFileHeader; //bmp文件头  
    12. BITMAPINFOHEADER bmpInfoHeader; //bmp信息头  
    13. RGBQUAD *pColorTable;       //bmp颜色表  
    14. unsigned char *pBmpData;    //bmp位图数据     
    15. unsigned char *pXBmpData;  //水平镜像bmp位图数据  
    16. unsigned char *pYBmpData;  //垂直镜像bmp位图数据  
    17. /** 
    18. * 函数名: readBmp 
    19. * 参  数: fileName -- 指向文件名的指针 
    20. * 功  能: 读取将要处理的图片的信息,成功返回TRUE 
    21. */  
    22. bool readBmp(char *fileName)  
    23. {  
    24.     FILE *fp = fopen(fileName,"rb");  //以二进制读方式打开  
    25.     if (NULL == fp)  
    26.     {     
    27.         cout<<"open failure!"<<endl;  
    28.         return FALSE;  
    29.     }  
    30.     //获得图片数据  
    31.     fread(&bmpFileHeader,sizeof(BITMAPFILEHEADER),1,fp);  
    32.     fread(&bmpInfoHeader,sizeof(BITMAPINFOHEADER),1,fp);  
    33.     pColorTable = new RGBQUAD[256];  
    34.     fread(pColorTable,sizeof(RGBQUAD),256,fp);  
    35.     int imgSize = bmpInfoHeader.biSizeImage;  
    36.     pBmpData = new unsigned char[imgSize];  
    37.     //因为大小没有改变,所以一起处理了  
    38.     pXBmpData = new unsigned char[imgSize];  
    39.     pYBmpData = new unsigned char[imgSize];  
    40.     fread(pBmpData,sizeof(unsigned char),imgSize,fp);  
    41.     fclose(fp); //关闭文件  
    42.     return TRUE;  
    43. }  
    44. /** 
    45. * 函数名: mirror 
    46. * 功  能: 对图片进行水平和垂直镜像操作 
    47. */  
    48. void mirror()  
    49. {  
    50.     int height = bmpInfoHeader.biHeight;  
    51.     int width = bmpInfoHeader.biWidth;  
    52.     int imgSize = bmpInfoHeader.biSizeImage;  
    53.     memset(pXBmpData,0,sizeof(unsigned char )*imgSize);  
    54.     memset(pYBmpData,0,sizeof(unsigned char )*imgSize);  
    55.     int lineByte = (width * 8 + 31) / 32 * 4;  //每行像素的字节数  
    56.     for(int i = 0; i < height; i++ )  
    57.     {  
    58.         for(int j = 0; j < width; j++ )  
    59.         {  
    60.             *(pXBmpData + i*lineByte + width - 1 - j) = *(pBmpData + i*lineByte + j); //水平镜像  
    61.             *(pYBmpData + (height - i - 1)*lineByte + j) = *(pBmpData + i*lineByte + j);  //垂直镜像  
    62.         }  
    63.     }  
    64. }  
    65. /** 
    66. * 函数名: writeBmp 
    67. * 参  数: fileName -- 处理完之后的bmp图像 
    68. * 功  能: 写入文件数据到相应的文件中 
    69. */  
    70. bool writeBmp(char *fileName,unsigned char *bmpData)  
    71. {  
    72.     FILE *fp = fopen(fileName,"wb"); //以二进制写方式打开  
    73.     if (NULL == fp)  
    74.     {  
    75.         cout<<"open failure!"<<endl;  
    76.         return FALSE;  
    77.     }  
    78.     int imgSize = bmpInfoHeader.biSizeImage;  
    79.     //写入数据  
    80.     fwrite(&bmpFileHeader,sizeof(BITMAPFILEHEADER),1,fp);  
    81.     fwrite(&bmpInfoHeader,sizeof(BITMAPINFOHEADER),1,fp);  
    82.     fwrite(pColorTable,sizeof(RGBQUAD),256,fp);  
    83.     fwrite(bmpData,sizeof(unsigned char),imgSize,fp);  
    84.     fclose(fp);  //关闭文件  
    85.     return TRUE;  
    86. }  
    87. /** 
    88. * 函数名: work 
    89. * 功  能: 主要处理 
    90. */  
    91. void work()  
    92. {  
    93.     char readFileName[] = "test.bmp";  
    94.     if (!readBmp(readFileName))  
    95.     {  
    96.         cout<<"read failure!"<<endl;  
    97.         return ;  
    98.     }  
    99.     mirror();  
    100.     char writeFileNameX[] = "X.bmp";  
    101.     char writeFileNameY[] = "Y.bmp";  
    102.     if (!writeBmp(writeFileNameX,pXBmpData))  
    103.     {  
    104.         cout<<"X write failure!"<<endl;  
    105.         return ;  
    106.     }  
    107.     if (!writeBmp(writeFileNameY,pYBmpData))  
    108.     {  
    109.         cout<<"Y write failure!"<<endl;  
    110.         return ;  
    111.     }  
    112.     //释放  
    113.     delete []pColorTable;  
    114.     delete []pBmpData;  
    115.     delete []pXBmpData;  
    116.     delete []pYBmpData;  
    117.     cout<<"mirror success!"<<endl;  
    118.   
    119. }  
    120. int main()  
    121. {  
    122.     work();  
    123.     return 0;  
    124. }  
上一篇:妙到毫巅,在阿里云容器服务中体验RAPIDS加速数据科学


下一篇:DockOne微信分享(六十八):应用容器env化实战