图像验证码识别(四)——灰度化和二值化

一、灰度化

灰度化应用很广,而且也比较简单。灰度图就是将白与黑中间的颜色等分为若干等级,绝大多数位256阶。在RGB模型种,黑色(R=G=B=0)与白色(R=G=B=255),那么256阶的灰度划分就是R=G=B=i,其中i取0到255.

从前面可以知道,OpenCV读取图片之后图像的颜色数据矩阵默认是3通道的,也就是RGB模型,所以每个pixel都有3个分量,分别代表r,g和b的值。因此将三个分量值都改为同一个灰度值,图片就实现灰度化。

灰度化的方法一般有以下几种:

1. 分量法 

在rgb三个分量种按照需求选取一个分量作为灰度值

2. 最大值 

选取rgb的最大值作为该pixel的灰度值

3. 平均值 

g[i,j] = (r[i,j] + g[i,j] + b[i,j]) / 3,取rgb的平均值作为灰度值

4. 加权变换 

由于人眼对绿色的敏感最高,对蓝色敏感最低,因此,按下式对RGB三分量进行加权平均能得到较合理的灰度图像。

g[i,j] = 0.3*r[i,j] + 0.59*g[i,j] + 0.11*b[i,j]

而OpenCV提供了灰度化的API,这里就直接调用了,采用的是加权变换的方法.

[cpp]  view plain  copy
  1. void Image::toGray()  
  2. {  
  3.     Mat iGray;  
  4.     if (getChannel() == 3)  
  5.     {  
  6.         std::cout << "width : " << getWidth() << std::endl;  
  7.         std::cout << "height: " << getHeight() << std::endl;  
  8.         cvtColor(m_Mat,iGray,COLOR_BGR2GRAY);  
  9.         m_Mat = iGray;  
  10.         std::cout << "The file has been converted to gray map successfully" << std::endl;  
  11.         std::cout << "***********************************" << std::endl;  
  12.     }  
  13.     else  
  14.     {  
  15.         std::cout << "the image file is not RGB file!" << std::endl;  
  16.     }  
  17. }  

二、二值化


二值化故名思议,就是整个图像所有像素只有两个值可以选择,一个是黑(灰度为0),一个是白(灰度为255)。二值化的好处就是将图片上的有用信息和无用信息区分开来,比如二值化之后的验证码图片,验证码像素为黑色,背景和干扰点为白色,这样后面对验证码像素处理的时候就会很方便。
常见的二值化方法为固定阀值和自适应阀值,固定阀值就是制定一个固定的数值作为分界点,大于这个阀值的像素就设为255,小于该阀值就设为0,这种方法简单粗暴,但是效果不一定好.另外就是自适应阀值,每次根据图片的灰度情况找合适的阀值。自适应阀值的方法有很多,这里采用了一种类似K均值的方法,就是先选择一个值作为阀值,统计大于这个阀值的所有像素的灰度平均值和小于这个阀值的所有像素的灰度平均值,再求这两个值的平均值作为新的阀值。重复上面的计算,直到每次更新阀值后,大于该阀值和小于该阀值的像素数目不变为止。
代码如下:
[cpp]  view plain  copy
  1. //k-mean method to find a threshold by itself  
  2. void Image::Binarization()  
  3. {  
  4.     int i,j,nWidth,nHeight;  
  5.     int nBack_count,nData_count;  
  6.     int nBack_sum,nData_sum;  
  7.     uchar ucThre,ucThre_new;  
  8.     int nValue;  
  9.   
  10.     nWidth = getWidth();  
  11.     nHeight = getHeight();  
  12.     //initial the threshold  
  13.     ucThre = 0;  
  14.     ucThre_new = 127;  
  15.   
  16.     std::cout << "Initial Threshold is :" << (int)ucThre_new << std::endl;  
  17.   
  18.     std::cout << "***********************************" << std::endl;  
  19.     while(ucThre != ucThre_new) {  
  20.         nBack_sum = nData_sum = 0;  
  21.         nBack_count = nData_count = 0;  
  22.   
  23.         for (j = 0; j < nHeight; ++j)  
  24.         {  
  25.             for (i = 0; i < nWidth; ++i)  
  26.             {  
  27.                 nValue = getPixel(i,j);  
  28.                 if (nValue > ucThre_new)  
  29.                 {  
  30.                     nBack_sum += nValue;  
  31.                     nBack_count++;  
  32.                 }  
  33.                 else {  
  34.                     nData_sum += nValue;  
  35.                     nData_count++;  
  36.                 }  
  37.             }  
  38.         }// end for i  
  39.   
  40.         nBack_sum = nBack_sum / nBack_count;  
  41.         nData_sum = nData_sum / nData_count;  
  42.         ucThre = ucThre_new;  
  43.         ucThre_new = (nBack_sum + nData_sum) / 2;  
  44.     }// end while  
  45.   
  46.     std::cout << "After Binarization threshold is :" << (int)ucThre_new << std::endl;  
  47.   
  48.     int nBlack = 0;  
  49.     int nWhite = 0;  
  50.   
  51.     for (j = 0; j < nHeight; ++j)  
  52.     {  
  53.         for (i = 0; i < nWidth; ++i)  
  54.         {  
  55.             nValue = getPixel(i,j);  
  56.             if (nValue > ucThre_new)  
  57.             {  
  58.                 setPixel(i,j,(uchar)WHITE);  
  59.                 nWhite++;  
  60.             }  
  61.             else {  
  62.                 setPixel(i,j,BLACK);  
  63.                 nBlack++;  
  64.             }  
  65.         }  
  66.     }  
  67.   
  68.     //backgroud is black,swap black and white  
  69.     if (nBlack > nWhite)  
  70.     {  
  71.         for (j = 0; j < nHeight; ++j)  
  72.             for (i = 0; i < nWidth; ++i)  
  73.             {  
  74.                 nValue = getPixel(i,j);  
  75.                 if ( !nValue )  
  76.                     setPixel(i,j,(uchar)WHITE);  
  77.                 else  
  78.                     setPixel(i,j,BLACK);  
  79.             }  
  80.     }  
  81.   
  82.     std::cout << "Binarization finished!" << std::endl;  
  83. }  

这里最后在二值化之后有一个反色的操作,在项目种规定了白色是背景,黑色是数据,但是有时候如果验证码图片的色彩比较重,最后会导致背景为黑色,数据为白色,所以这里进行了一个判断,由于绝大多数情况下,验证码的大部分像素还是充当背景的,所以如果最后统计的黑色像素点数目大于白色的,就将所有像素的二值取反。

你可能感兴趣的:(验证码识别)