MFC图像点运算之灰度线性变化、灰度非线性变化、阈值化和均衡化处理

 本文主要讲述基于VC++6.0 MFC图像处理的应用知识,主要结合自己大三所学课程《数字图像处理》及课件进行讲解,主要通过MFC单文档视图实现显示BMP图片点运算处理,包括图像灰度线性变换、灰度非线性变换、图像阈值化处理、图像均衡化处理等知识,并结合前一篇论文灰度直方图进行展示 。同时文章比较详细基础,希望该篇文章对你有所帮助,尤其是初学者和学习图像处理的学生。
       【数字图像处理】一.MFC详解显示BMP格式图片
       【数字图像处理】二.MFC单文档分割窗口显示图片
       【数字图像处理】三.MFC实现图像灰度、采样和量化功能详解
       【数字图像处理】四.MFC对话框绘制灰度直方图
        免费资源下载地址:
        http://download.csdn.net/detail/eastmount/8764373

 

一. 点运算与初始操作

        图像的点运算是图像处理中非常基础的技术,它主要用于改变一篇图像的灰度分布范围,通过一定的变换函数将图像的像素进行转换,最终生成一幅新的图像。点运算的最大特点就是输出像素值只与当前输入像素值相关。定义如下。
        点运算(Point Operation)指对于一幅输入图像,将产生一幅输出图像,输出图像的每个像素点的灰度值由输入像素点决定。
        点运算由灰度变换函数(Grap Scale Transformation,GST)确定:B(x,y)=F[A(x,y)]
        需要注意一下几点:
        (1).与局部或邻域运算的差别,输入像素和输出像素是一一对应的;(2).与几何运算的差别,不改变图像的空间关系;(3).又称为对比增强,对比拉伸或灰度变换。

        在前面第四篇博客的基础上增加点运算处理。
        第一步:在资源视图中Menu中添加“图像点运算”菜单栏如下所示:


        对应的ID值为:
        线性变换 ID_DYS_XXYD(点运算 线性移动) ID_DYS_XXZQ( 点运算 线性增强)
                       ID_DYS_XXJX(点运算 线性减小)  ID_DYS_XXQB(点运算 线性求补)
        非线性变换 ID_DYS_FXXPF(点运算 非线性平方) ID_DYS_FXXHS(非线性函数)
        阈值变换 ID_DYS_YZBH(点运算 阈值变换) 图像均衡化 ID_DYS_JHH

        第二步:打开类向导(Ctrl+W),为点运算每个ID菜单添加相应的功能处理函数,如下图所示:选择类CImageProcessingView,在选择IDs为ID_DYS_...(点运算)添加函数OnDysXxqb()线性求补。

 

 

二. 线性变换

        图像线性变换是通过建立灰度映射来调整资源图像的灰度,从而达到图像增强的目的。其中GST函数f(D)为线性的,即:


        若a=1,b=0图像像素不发生变化
        若a=1,b!=0图像所有灰度值上移或下移
        若a>1输出图像对比度增强
        若0<a<1输出图像对比度减小
        若a<0暗区域变亮,亮区域变暗,图像求补


 

        1.D(B)=D(A)+50
        首先是图像移动,代码如下:

 
  1. /**********************************************************************/

  2. /* 图像点运算 4种线性变化直方图:

  3. /* ID_DYS_XXYD:表示线性灰度变化移动 D(B)=D(A)+50 灰度值上移下移

  4. /* ID_DYS_XXZQ:表示线性灰度变化增强 D(B)=1.5*D(A) 图像对比度增强

  5. /* ID_DYS_XXJX:表示线性灰度变化减小 D(B)=0.8*D(A) 图像对比度减小

  6. /* ID_DYS_XXQB:表示线性灰度求补 D(B)=-1*D(A)+255 图像暗区变亮,亮区变暗

  7. /**********************************************************************/

  8.  
  9. // 1.点运算 线性灰度变化移动 D(B)=D(A)+50

  10. void CImageProcessingView::OnDysXxyd()

  11. {

  12. // TODO: Add your command handler code here

  13. if(numPicture==0) {

  14. AfxMessageBox("载入图片后才能线性灰度运算!",MB_OK,0);

  15. return;

  16. }

  17. AfxMessageBox("线性灰度直方图-灰度变化移动 D(B)=D(A)+50!",MB_OK,0);

  18. int i;

  19. //打开临时的图片

  20. FILE *fpo = fopen(BmpName,"rb");

  21. FILE *fpw = fopen(BmpNameLin,"wb+");

  22. //读取文件

  23. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  24. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  25. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  26. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  27. //灰度图像

  28. unsigned char color;

  29. unsigned char red,green,blue;

  30. for( i=0; i<m_nImage/3; i++ )

  31. {

  32. fread(&red,sizeof(char),1,fpo);

  33. fread(&green,sizeof(char),1,fpo);

  34. fread(&blue,sizeof(char),1,fpo);

  35.  
  36. if( (int)red+50 >255 )

  37. red=255;

  38. else

  39. red=(int)red+50;

  40.  
  41. if( (int)green+50>255 )

  42. green=255;

  43. else

  44. green=(int)green+50;

  45.  
  46. if( (int)blue+50>255 )

  47. blue=255;

  48. else

  49. blue=(int)blue+50;

  50.  
  51. fwrite(&red,sizeof(char),1,fpw);

  52. fwrite(&green,sizeof(char),1,fpw);

  53. fwrite(&blue,sizeof(char),1,fpw);

  54. }

  55. fclose(fpo);

  56. fclose(fpw);

  57. numPicture = 2;

  58. level=101; //赋值101在ShowBitmap中调用显示处理后的图片

  59. Invalidate();

  60. }

        同时修改void CImageProcessingView::ShowBitmap(CDC *pDC, 
CString BmpName)函数中的代码:

 
  1. else //图像点运算 线性变化

  2. if(level=101)

  3. {

  4. m_hBitmapChange = (HBITMAP) LoadImage(NULL,BmpNameLin,IMAGE_BITMAP,0,0,

  5. LR_LOADFROMFILE|LR_DEFAULTSIZE|LR_CREATEDIBSECTION);

  6. }

        运行效果如下图所示,同时我截取了直方图(RGB相同只显示一种)。

        可以发现图像的灰度上移了50,图像更白了(黑0-255白)。

        2.D(B)=1.5*D(A)

 
  1. // 2.点运算 线性灰度变化增强 D(B)=1.5*D(A)

  2. void CImageProcessingView::OnDysXxzq()

  3. {

  4. if(numPicture==0) {

  5. AfxMessageBox("载入图片后才能线性灰度运算!",MB_OK,0);

  6. return;

  7. }

  8. AfxMessageBox("线性灰度直方图-灰度变化增强 D(B)=1.5*D(A)!",MB_OK,0);

  9. int i;

  10. //打开临时的图片

  11. FILE *fpo = fopen(BmpName,"rb");

  12. FILE *fpw = fopen(BmpNameLin,"wb+");

  13. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  14. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  15. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  16. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  17. //灰度图像

  18. unsigned char color;

  19. unsigned char red,green,blue;

  20. for( i=0; i<m_nImage/3; i++ )

  21. {

  22. fread(&red,sizeof(char),1,fpo);

  23. fread(&green,sizeof(char),1,fpo);

  24. fread(&blue,sizeof(char),1,fpo);

  25.  
  26. if( (int)red*1.5 >255 )

  27. red=255;

  28. else

  29. red=(int)red*1.5;

  30.  
  31. if( (int)green*1.5>255 )

  32. green=255;

  33. else

  34. green=(int)green*1.5;

  35.  
  36. if( (int)blue*1.5>255 )

  37. blue=255;

  38. else

  39. blue=(int)blue*1.5;

  40.  
  41. fwrite(&red,sizeof(char),1,fpw);

  42. fwrite(&green,sizeof(char),1,fpw);

  43. fwrite(&blue,sizeof(char),1,fpw);

  44. }

  45. fclose(fpo);

  46. fclose(fpw);

  47. numPicture = 2;

  48. level=101; //线性变化 ShowBitmap中调用

  49. Invalidate();

  50. }

        运行效果如下图所示,图像对比度增强,平均灰度122*1.5=181

 

 

        3.D(B)=0.8*D(A)

 
  1. // 3.点运算 线性灰度变化减小D(B)=0.8*D(A)

  2. void CImageProcessingView::OnDysXxjx()

  3. {

  4. if(numPicture==0) {

  5. AfxMessageBox("载入图片后才能线性灰度处理!",MB_OK,0);

  6. return;

  7. }

  8. AfxMessageBox("线性灰度直方图-灰度减小 D(B)=0.8*D(A)!",MB_OK,0);

  9. int i;

  10. //打开临时的图片

  11. FILE *fpo = fopen(BmpName,"rb");

  12. FILE *fpw = fopen(BmpNameLin,"wb+");

  13. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  14. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  15. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  16. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  17. //灰度图像

  18. unsigned char color;

  19. unsigned char red,green,blue;

  20. for( i=0; i<m_nImage/3; i++ )

  21. {

  22. fread(&red,sizeof(char),1,fpo);

  23. fread(&green,sizeof(char),1,fpo);

  24. fread(&blue,sizeof(char),1,fpo);

  25.  
  26. red=(int)red*0.8;

  27. green=(int)green*0.8;

  28. blue=(int)blue*0.8;

  29.  
  30. fwrite(&red,sizeof(char),1,fpw);

  31. fwrite(&green,sizeof(char),1,fpw);

  32. fwrite(&blue,sizeof(char),1,fpw);

  33. }

  34. fclose(fpo);

  35. fclose(fpw);

  36. numPicture = 2;

  37. level=101;

  38. Invalidate();

  39. }

        运行如下图所示,图像减弱。



        4.D(B)=-1*D(A)+255

 
  1. // 4.点运算 线性灰度求补 D(B)=-1*D(A)+255

  2. void CImageProcessingView::OnDysXxqb()

  3. {

  4. if(numPicture==0) {

  5. AfxMessageBox("载入图片后才能线性灰度处理!",MB_OK,0);

  6. return;

  7. }

  8. AfxMessageBox("线性灰度直方图-灰度求补 D(B)=-1*D(A)+255!",MB_OK,0);

  9. int i;

  10. //打开临时的图片

  11. FILE *fpo = fopen(BmpName,"rb");

  12. FILE *fpw = fopen(BmpNameLin,"wb+");

  13. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  14. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  15. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  16. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  17. //灰度图像

  18. unsigned char color;

  19. unsigned char red,green,blue;

  20. for( i=0; i<m_nImage/3; i++ )

  21. {

  22. fread(&red,sizeof(char),1,fpo);

  23. fread(&green,sizeof(char),1,fpo);

  24. fread(&blue,sizeof(char),1,fpo);

  25.  
  26. red=(int)red*(-1)+255;

  27. green=(int)green*(-1)+255;

  28. blue=(int)blue*(-1)+255;

  29.  
  30. fwrite(&red,sizeof(char),1,fpw);

  31. fwrite(&green,sizeof(char),1,fpw);

  32. fwrite(&blue,sizeof(char),1,fpw);

  33. }

  34. fclose(fpo);

  35. fclose(fpw);

  36. numPicture = 2;

  37. level=101;

  38. Invalidate();

  39. }

        运行效果如下图所示,它是图像的求补,发现直方图是互补的。

        PS:注意图片下面的直方图应该还有一个处理后的直方图,但原理都一样,我不想重复工作,你自己可以去简单实现下,参考第四篇文章。同时这些图片制作还挺麻烦的,只是为了给你更好的呈现它们的变化,希望对你有用和尊重作者,不喜勿喷~

 

 

三. 非线性变换

        灰度非线性变换主要包括对数变换、幂次变换、指数变换、分段函数变换,通过非线性关系对图像进行灰度处理,下面主要讲解课件中的两个函数对其进行处理。其中对数变换实现了扩展低灰度值而压缩高灰度值的效果,图像灰度分布更符合而你的视觉特征。


        1.D(B)=D(A)*D(A)/252

 
  1. /************************************************************************/

  2. /* 2种非线性变化直方图:

  3. /* ID_DYS_FXXPF:表示非线性平方灰度变化,D(B)=D(A)*D(A)/255

  4. /* ID_DYS_FXXHS:表示非线性函数灰度变化,D(B)=D(A)+0.8*D(A)*(255-D(A))/255

  5. /************************************************************************/

  6.  
  7. // 非线性平方灰度变化 D(B)=D(A)*D(A)/252

  8. void CImageProcessingView::OnDysFxxpf()

  9. {

  10. if(numPicture==0)

  11. {

  12. AfxMessageBox("载入图片后才能非线性灰度处理!",MB_OK,0);

  13. return;

  14. }

  15. AfxMessageBox("非线性灰度变化 D(B)=D(A)*D(A)/255!",MB_OK,0);

  16. int i;

  17. //打开临时的图片

  18. FILE *fpo = fopen(BmpName,"rb");

  19. FILE *fpw = fopen(BmpNameLin,"wb+");

  20. //读取文件

  21. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  22. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  23. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  24. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  25. //灰度图像

  26. unsigned char color;

  27. unsigned char red,green,blue;

  28. for( i=0; i<m_nImage/3; i++ )

  29. {

  30. fread(&red,sizeof(char),1,fpo);

  31. fread(&green,sizeof(char),1,fpo);

  32. fread(&blue,sizeof(char),1,fpo);

  33.  
  34. red=(int)red*(int)red/255;

  35. green=(int)green*(int)green/255;

  36. blue=(int)blue*(int)blue/255;

  37.  
  38. fwrite(&red,sizeof(char),1,fpw);

  39. fwrite(&green,sizeof(char),1,fpw);

  40. fwrite(&blue,sizeof(char),1,fpw);

  41. }

  42. fclose(fpo);

  43. fclose(fpw);

  44. numPicture = 2;

  45. level=101;

  46. Invalidate();

  47. }

        运行效果如下图所示:


        2.D(B)=D(A)+0.8*D(A)*(255-D(A))/255

 
  1. // 非线性函数灰度变化 D(B)=D(A)+0.8*D(A)*(255-D(A))/255

  2. void CImageProcessingView::OnDysFxxhs()

  3. {

  4. if(numPicture==0)

  5. {

  6. AfxMessageBox("载入图片后才能非线性灰度处理!",MB_OK,0);

  7. return;

  8. }

  9. AfxMessageBox("线性灰度直方图-灰度变化增强 D(B)=D(A)+0.8*D(A)*(255-D(A))/255!",MB_OK,0);

  10. int i;

  11.  
  12. FILE *fpo = fopen(BmpName,"rb");

  13. FILE *fpw = fopen(BmpNameLin,"wb+");

  14. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  15. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  16. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  17. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  18.  
  19. unsigned char color;

  20. unsigned char red,green,blue;

  21. for( i=0; i<m_nImage/3; i++ )

  22. {

  23. fread(&red,sizeof(char),1,fpo);

  24. fread(&green,sizeof(char),1,fpo);

  25. fread(&blue,sizeof(char),1,fpo);

  26.  
  27. if( ((int)red+0.8*(int)red*(255-(int)red)/255) > 255 )

  28. red=255;

  29. else

  30. red=(int)red+0.8*(int)red*(255-(int)red)/255;

  31.  
  32. if( ((int)green+0.8*(int)green*(255-(int)green)/255) > 255 )

  33. green=255;

  34. else

  35. green=(int)green+0.8*(int)green*(255-(int)green)/255;

  36.  
  37. if( ((int)blue+0.8*(int)blue*(255-(int)blue)/255) > 255 )

  38. blue=255;

  39. else

  40. blue=(int)blue+0.8*(int)blue*(255-(int)blue)/255;

  41.  
  42. fwrite(&red,sizeof(char),1,fpw);

  43. fwrite(&green,sizeof(char),1,fpw);

  44. fwrite(&blue,sizeof(char),1,fpw);

  45. }

  46. fclose(fpo);

  47. fclose(fpw);

  48. numPicture = 2;

  49. level=101;

  50. Invalidate();

  51. }

        运行效果如下图所示:


        写到此处你会发现图像灰度的线性变换和非线性变换是非常简单的,主要是通过以下步骤完成:
        第一步:赋值处理后图像的BMP头信息
            FILE *fpo = fopen(BmpName,"rb");
            FILE *fpw = fopen(BmpNameLin,"wb+");
            fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);
            fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);
            fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);
            fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);
        第二步:通过循环和线性变换或非线性便函函数处理每一个像素
            for( i=0; i<m_nImage/3; i++ )
            {
                 fread(&red,sizeof(char),1,fpo);
                 处理像素RBG 如:red=(int)red*(int)red/255;
                 fwrite(&red,sizeof(char),1,fpw);
            }
         第三步:调用ShowBitmap自定义函数并重绘图像
            numPicture = 2;
            level=101;
            Invalidate();
        而它的主要应用包括:光度学标定,希望数字图像的灰度能够真实反映图像的物理特性;对比度增强和对比度扩展;显示标定和轮廓线确定(阈值化)。

 

 

四. 灰度阈值化

        阈值又称为临界值,它的目的是确定出一个范围,然后这个范围内的部分使用同一种方法处理,而阈值之外的部分则使用另一种处理方法或保持原样。常用的包括产生二值图:当x<T时y=0,当x>=T时y=255(其中T是阈值)。阈值变换在生物学上的应用比较广泛,常用语细胞图像分割等。
        打开类向导(Ctrl+W)生成选择ImageProcessingView类,IDs选择ID_DYS_YZBH后添加相应的函数。代码如下:

 
  1. /**************************************************************/

  2. /* ID_DYS_YZBH:表示点运算阈值变换 也看做灰度拉伸

  3. /* 此处的拉伸是:阈值化(thresholding)可以看作是削波的一个特例

  4. /* 只要令削波中的g1old=g2old就实现了阈值化。

  5. /* 阈值就象个门槛,比它大就是白,比它小就是黑,二值

  6. /**************************************************************/

  7.  
  8. void CImageProcessingView::OnDysYzbh()

  9. {

  10. if(numPicture==0)

  11. {

  12. AfxMessageBox("载入图片后才能点运算阈值化处理!",MB_OK,0);

  13. return;

  14. }

  15. AfxMessageBox("图像点运算阈值化处理!",MB_OK,0);

  16. //读写文件

  17. FILE *fpo = fopen(BmpName,"rb");

  18. FILE *fpw = fopen(BmpNameLin,"wb+");

  19. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  20. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  21. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  22. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  23. //处理

  24. unsigned char color;

  25. unsigned char red,green,blue;

  26. for(int i=0; i<m_nImage/3; i++ )

  27. {

  28. fread(&red,sizeof(char),1,fpo);

  29. fread(&green,sizeof(char),1,fpo);

  30. fread(&blue,sizeof(char),1,fpo);

  31.  
  32. if( (int)red > 128 )

  33. red=255;

  34. else

  35. red=0;

  36.  
  37. if( (int)green > 128 )

  38. green=255;

  39. else

  40. green=0;

  41.  
  42. if( (int)blue > 128 )

  43. blue=255;

  44. else

  45. blue=0;

  46.  
  47. fwrite(&red,sizeof(char),1,fpw);

  48. fwrite(&green,sizeof(char),1,fpw);

  49. fwrite(&blue,sizeof(char),1,fpw);

  50. }

  51. fclose(fpo);

  52. fclose(fpw);

  53. numPicture = 2;

  54. level=101;

  55. Invalidate();

  56. }

        运行效果如下图所示,感觉还挺好看的,显然此时的直方图就是0和255两条直线。

 

 

五. 灰度均衡化

        灰度均衡化的目的是使一输入图像转换为在每一灰度级上都有相同的像素点(即输出的直方图是平的),它可以产生一幅灰度级分布概率均衡的图像。
        换句话说,经过均衡化后的图像在每一级灰度上像素点的数量相差不大,对应的灰度直方图的每一级高度也相差不大。它是增强图像的有效手段之一。
        研究思路是通过直方图变换公式实现:


        它的步骤如下图所示:

        例:有一幅图象,共有16级灰度,其直方图分布为Pi, i=0,1,…,15,求经直方图均衡化后,量化级别为10级的灰度图象的直方图分布Qi,其中Pi和Qi为分布的概率,即灰度i出现的次数与总的点数之比。
        Pi:0.03, 0, 0.06, 0.10, 0.20, 0.11, 0, 0, 0, 0.03, 0, 0.06, 0.10, 0.20, 0.11, 0
        步骤1:用一个数组s记录Pi,即s[0]=0.03,s[1]=0,s[2]=0.06,…,s[14]=0.11,s[15]=0
        步骤2:i从1开始,令s[i]=s[i]+s[i-1],得到的结果是s: 0.03,  0.03, 0.09,  0.19,  0.39, 0.50,  0.50,  0.50, 0.50,  0.53,  0.53, 0.59,  0.69,  0.89, 1.0,  1.0
        步骤3:用一个数组L记录新的调色板索引值,即令L[i]=s[i]×(10-1),得到的结果是L:0,0,1,2,4,5,5,5,5,5,5,5,6,8,9,9
        这样就找到了原来的调色板索引值和新的调色板索引值之间的对应关系,即
        0→0,  1→0, 2→1,  3→2,  4→4, 5→5,  6→5,  7→5, 8→5,  9→5,  10→5, 11→5,  12→6,  13→8, 14→9,  15→9。
       步骤4:将老的索引值对应的概率合并,作为对应的新的索引值的概率。例如,原来的索引值0,1都对应了新的索引值0,则灰度索引值为0的概率为P0+P1=0.03;新的索引值3和7找不到老的索引值与之对应,所以令Q3和Q7为0。最后得到的结果是Qi:0.03,  0.06, 0.10,  0,  0.20, 0.20,  0.10,  0, 0.20,  0.11 

        代码中有详细注释如下:

 
  1. // ID_DYS_JHH:表示图像均衡化 相见算法

  2. void CImageProcessingView::OnDysJhh()

  3. {

  4. if(numPicture==0) {

  5. AfxMessageBox("载入图片后才能图像均衡化!",MB_OK,0);

  6. return;

  7. }

  8. AfxMessageBox("图像均衡化!",MB_OK,0);

  9.  
  10. //第一步:获取图像的数据信息

  11. //此操作可以在打开图片时就进行 在直方图采样(ZFTCY)中也有该代码

  12. FILE *fpo = fopen(BmpName,"rb");

  13. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  14. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  15.  
  16. int i,j,k;

  17. for(j=0;j<256;j++) { //定义数组并清零

  18. Red[j]=0;

  19. Green[j]=0;

  20. Blue[j]=0;

  21. }

  22.  
  23. //计算4个数据

  24. unsigned char red,green,blue;

  25. int IntRed,IntGreen,IntBlue; //强制转换

  26. double sumRedHD=0,sumGreenHD=0,sumBlueHD=0; //记录像素总的灰度值和

  27. for(i=0; i<m_nImage/3; i++ )

  28. {

  29. fread(&red,sizeof(char),1,fpo);

  30. IntRed=int(red);

  31. sumRedHD=sumRedHD+IntRed;

  32. if( IntRed>=0 && IntRed<256 ) Red[IntRed]++;

  33.  
  34. fread(&green,sizeof(char),1,fpo);

  35. IntGreen=int(green);

  36. sumGreenHD=sumGreenHD+IntGreen;

  37. if( IntGreen>=0 && IntGreen<256 ) Green[IntGreen]++;

  38.  
  39. fread(&blue,sizeof(char),1,fpo);

  40. IntBlue=int(blue);

  41. sumBlueHD=sumBlueHD+IntBlue;

  42. if( IntBlue>=0 && IntBlue<256 ) Blue[IntBlue]++;

  43. }

  44. fclose(fpo);

  45.  
  46. /*****************************************************************/

  47. /* 图像均衡化处理

  48. /* 利用全局变量 Red[256] Blue[256] Green[256]

  49. /* 第一步:用3个数组Count..记录0-255灰度出现的概率,即

  50. /* 概率=该灰度出现次数*3/总得像素 (因为分成3部分RGB)

  51. /* 第二步:i从1开始,令s[i]=s[i]+s[i-1] 记录新概率数

  52. /* 第三步:用一个数组L记录新的调色板索引值,即

  53. /* L[i]=s[i]×(256-1)结果四舍五入2.8即为3

  54. /* 第四步:将老的索引值对应的概率合并,作为对应的新的索引值的概率

  55. /* 1.原来的索引值0,1都对应了新的索引值0,则灰度索引值为0的概率

  56. /* 为P0+P1=0.03

  57. /* 2.新的索引值3和7找不到老的索引值与之对应,所以令Q3和Q7为0

  58. /*****************************************************************/

  59.  
  60. //记录出现的概率,会加到1 用于相加到调色板

  61. float CountRed[256],CountGreen[256],CountBlue[256];

  62. //记录原始数据,不会相加到1 用于计算新灰度概率

  63. float CountRedLin[256],CountGreenLin[256],CountBlueLin[256];

  64.  
  65. for( k=0 ; k<256 ; k++ )

  66. {

  67. CountRed[k]=(float)(Red[k])*3/m_nImage;

  68. CountRedLin[k]=CountRed[k];

  69. CountGreen[k]=(float)(Green[k])*3/m_nImage;

  70. CountGreenLin[k]=CountGreen[k];

  71. CountBlue[k]=(float)(Blue[k])*3/m_nImage;

  72. CountBlueLin[k]=CountBlue[k];

  73. }

  74.  
  75. for( k=1 ; k<256 ; k++ )

  76. {

  77. CountRed[k]=CountRed[k]+CountRed[k-1];

  78. CountGreen[k]=CountGreen[k]+CountGreen[k-1];

  79. CountBlue[k]=CountBlue[k]+CountBlue[k-1];

  80. }

  81.  
  82. /****************************************************/

  83. /* 此处百度到一个四舍五入浮点型的算法:

  84. /* float a=3.456; 保留到小数点后两位

  85. /* float b=(int)((a * 100) + 0.5) / 100.0;

  86. /* output b=3.46

  87. /****************************************************/

  88.  
  89. int LRed[256],LGreen[256],LBlue[256]; //记录调色板

  90. for( k=0 ; k<256 ; k++ )

  91. {

  92. LRed[k]=(int)(CountRed[k]*(256-1)+0.5);

  93. LGreen[k]=(int)(CountGreen[k]*(256-1)+0.5);

  94. LBlue[k]=(int)(CountBlue[k]*(256-1)+0.5);

  95. }

  96.  
  97. //第三步:处理均衡化图像写入 打开临时的图片

  98. fpo = fopen(BmpName,"rb");

  99. fread(&bfh,sizeof(BITMAPFILEHEADER),1,fpo);

  100. fread(&bih,sizeof(BITMAPINFOHEADER),1,fpo);

  101.  
  102. FILE *fpw = fopen(BmpNameLin,"wb+");

  103. fwrite(&bfh,sizeof(BITMAPFILEHEADER),1,fpw);

  104. fwrite(&bih,sizeof(BITMAPINFOHEADER),1,fpw);

  105.  
  106. //m_nWidth*m_nHeight 读取图片最后一行不为m_nWidth时会报错 改为m_nImage/3

  107. for( i=0; i<m_nImage/3 ; i++ )

  108. {

  109. fread(&red,sizeof(char),1,fpo);

  110. fread(&green,sizeof(char),1,fpo);

  111. fread(&blue,sizeof(char),1,fpo);

  112.  
  113. red=LRed[int(red)];

  114. green=LGreen[int(green)];

  115. blue=LBlue[int(blue)];

  116.  
  117. fwrite(&red,sizeof(char),1,fpw);

  118. fwrite(&green,sizeof(char),1,fpw);

  119. fwrite(&blue,sizeof(char),1,fpw);

  120. }

  121. fclose(fpw);

  122. numPicture = 2;

  123. level=101;

  124. Invalidate();

  125. }

        运行结果如下图所示,图像增强而且异常清晰:

 


        最后介绍下图像对比度拉伸,它就是把你感兴趣的灰度范围拉开,使得该范围内像素,亮的更亮,暗的更暗,从而达到增强对比度的目的。
        如下图所示,a、b、c为三段直线的斜率,g1old和g2old表示途中要进行对比度扩展的范围,g1new和g2new表示对应的新值。当g1old=g2old就是二值图像阈值化处理。

 

          由于灰度界别也是255这个约束,所以满足

       其中g1old=100,g2old=150,b=3.0的运行效果如下所示:

 


        

//

https://blog.csdn.net/eastmount/article/details/46312145

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

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

相关文章

马化腾:5G和AI双核驱动产业互联网进入“快车道”

来源&#xff1a;腾讯科技腾讯科技讯 3月30至31日&#xff0c;2019中国&#xff08;深圳&#xff09;IT领袖峰会以“IT新未来&#xff1a;5G与人工智能”为主题&#xff0c;汇聚了众多科技领袖和各界精英&#xff0c;聚焦未来通信、工业互联网、数字城市、金融科技等热门话题。…

MFC空间几何变换之图像平移、镜像、旋转、缩放

本文主要讲述基于VC6.0 MFC图像处理的应用知识&#xff0c;主要结合自己大三所学课程《数字图像处理》及课件进行讲解&#xff0c;主要通过MFC单文档视图实现显示BMP图片空间几何变换&#xff0c;包括图像平移、图形旋转、图像反转倒置镜像和图像缩放的知识。同时文章比较详细基…

超级干货:一文看懂5G产业链及投资机会

来源&#xff1a;新材料在线摘要&#xff1a;本文将讲述5G行业概况、产业链结构、上游关键原材料、本行业竞争格局及材料重点应用领域。报告合集涵盖5G关键材料、5G天线、氮化镓半导体、导热材料、电磁屏蔽材料、高频覆铜板基材、微波介质陶瓷、先进封装、手机外壳等九大市场研…

MFC图像增强之图像普通平滑、高斯平滑、Laplacian、Sobel、Prewitt锐化

本文主要讲述基于VC6.0 MFC图像处理的应用知识&#xff0c;主要结合自己大三所学课程《数字图像处理》及课件进行讲解&#xff0c;主要通过MFC单文档视图实现显示BMP图像增强处理&#xff0c;包括图像普通平滑、高斯平滑、不同算子的图像锐化知识。希望该篇文章对你有所帮助&am…

南京大学教授施斌及其团队—— 光纤变“神经” 大地能感知

来源&#xff1a;人民日报你能相信吗&#xff1f;一根头发丝粗细的光纤&#xff0c;根据不同地质环境和多场监测要求&#xff0c;穿上各种“定制”的外衣&#xff0c;就能变身敏感强健的“大地感知神经”&#xff0c;使得大地一有灾害异动&#xff0c;远在千里之外的监测系统就…

MFC详解显示BMP格式图片

本文主要是讲述《数字图像处理》系列栏目中的第一篇文章.主要详细介绍了BMP图片格式,同时使用C和MFC显示BMP格式,主要结合自己的《数字图像处理》课程和以前的项目叙述讲解. 一.BMP图片格式定义 BMP文件格式是Windows操作系统推荐和支持的标准图像文件格式,是一种将内存或显示…

0pencv——图像腐蚀

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("小狗1.jpg");imshow("显示图像", srcImage);Mat element getStructuringElement(MORPH_RECT, Size(…

腾讯研究院发布:《人工智能+制造产业发展研究》报告

来源&#xff1a;腾讯研究院摘要&#xff1a;工业革命以后的“自动化”概念追求的是机器自动生产&#xff0c;本质是“机器替人”&#xff0c;强调在完全不需要人的情况下进行不间断的大规模机器生产&#xff1b;而“智能化”追求的是机器的柔性生产&#xff0c;本质是“人机协…

Opencv——图像模糊

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("小狗1.jpg");imshow("原图像", srcImage);Mat dstImage;blur(srcImage, dstImage, Size(5, 5));imsh…

Android开发中依赖注入的应用

什么是依赖注入&#xff1f; 依赖是指一个对象持有其他对象的引用。依赖注入则是将这些依赖对象传递给被依赖对象&#xff0c;而不是被依赖对象自己创建这些对象。 public class MyClass{private AnotherClass mAnotherObject;public MyClass(){mAnotherObject new AnotherCla…

工业富联2018年报来了!上市后首张成绩单大起底

未来智能实验室是人工智能学家与科学院相关机构联合成立的人工智能&#xff0c;互联网和脑科学交叉研究机构。未来智能实验室的主要工作包括&#xff1a;建立AI智能系统智商评测体系&#xff0c;开展世界人工智能智商评测&#xff1b;开展互联网&#xff08;城市&#xff09;云…

Opencv——图像膨胀

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("小狗1.jpg");imshow("原图像", srcImage);Mat dstImage;Mat element getStructuringElement(MORPH_…

深度学习背后的基础-神经网络揭秘

来源&#xff1a;混沌巡洋舰摘要&#xff1a;最近&#xff0c; 深度学习三杰获得了计算机界最重要的图灵奖&#xff0c; 它们的贡献都集中在对深度学习的根据神经网络的理论突破。 今天我们看到的所有和人工智能有关的伟大成就&#xff0c; 从阿法狗到自动驾驶&#xff0c; 从海…

Opencv——灰度变换、直方图均衡化

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("lena.bmp");Mat grayImage;cvtColor(srcImage, grayImage, COLOR_BGR2GRAY);Mat dstImage;equalizeHist(grayIm…

并发模型之——共享内存模型(线程与锁)理论篇

这里我们使用Java的线程与锁来解析共享内存模型&#xff1b;做过java开发并且了解线程安全问题的知道&#xff0c;要使某段代码是线程安全的那必须要满足两个条件&#xff1a;内存可见性、原子性&#xff1b; 内存可见性 在JVM规定多个线程进行通讯是通过共享变量进行的&a…

谷歌员工怒了 900人联名抗议 刚成立的AI道德委员会处境尴尬

来源&#xff1a;网易智能谷歌员工又怒了。3月26日&#xff0c;谷歌宣布成立人工智能项目外部顾问委员会&#xff0c;该委员会将与谷歌就面部识别和公平性等人工智能的主要问题进行磋商。争议的焦点在于&#xff0c;谷歌将保守派传统基金会(Heritage Foundation)主席凯科尔斯詹…

Opencv——基于索引表的图像细化

图像细化针对的是二值图像 或者用阀值处理的二值图像。基于索引表的细化算法大致是遍历被二值化图像的边缘&#xff0c;根据边缘点的八连通域情况查找索引表以确定该边缘点是否能够被删除。根据一些细化规则我们可以建立索引表&#xff0c;因此我们的主要工作就是不断地遍历边…

DeepMind推出首个商业产品,30秒内准确诊断眼疾!

来源&#xff1a;Financial Times、智东西编译摘要&#xff1a;这个设备能像最好的医学专家一样&#xff0c;准确地诊断各种眼部疾病。4月1日&#xff0c;谷歌母公司Alphabet旗下位于伦敦的AI部门DeepMind已打造出了可诊断复杂眼部疾病的商业医疗设备原型&#xff0c;这将是Dee…

Opencv——Sobel边缘检测

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("lena.jpg");Mat dstImage_x, dstImage_y;Sobel(srcImage, dstImage_x, CV_8U, 1, 0);Sobel(srcImage, dstImag…

Head first servlet and jsp学习笔记

学习中遇到的问题&#xff1a;java基础不行&#xff0c;都忘光了。 主要是&#xff1a;继承&#xff0c;接口&#xff0c;多线程&#xff0c;IO。尤其是多线程&#xff0c;在分布式系统中应该使用的比较多 第一章&#xff1a;前言和体系结构 HTTP协议&#xff1a; TCP/IP的上层…