Java基于opencv实现图像数字识别(三)—灰度化和二值化

Java基于opencv实现图像数字识别(三)—灰度化和二值化

一、灰度化

灰度化:在RGB模型中,如果R=G=B时,则彩色表示灰度颜色,其中R=G=B的值叫灰度值;因此,灰度图像每个像素点只需一个字节存放灰度值(又称强度值、亮度值),灰度范围为0-255。一般常用的是加权平均法来求像素点的灰度值,opencv开发库所采用的一种求灰度值算法如下;
:)Gray = 0.072169 * B + 0.715160 * G + 0.212671 * R

有两种方式可以实现灰度化,如下

方式1
@Test
public void toGray() {
    // 这个必须要写,不写报java.lang.UnsatisfiedLinkError
    System.loadLibrary(Core.NATIVE_LIBRARY_NAME);

    File imgFile = new File("C:/Users/admin/Desktop/open/test.png");
    String dest = "C:/Users/admin/Desktop/open";
        
    //方式一
    Mat src = Imgcodecs.imread(imgFile.toString(), Imgcodecs.CV_LOAD_IMAGE_GRAYSCALE);
    //保存灰度化的图片
    Imgcodecs.imwrite(dest + "/toGray" + imgFile.getName(), src);
}
方式2
@Test
public void toGray() {
    // 这个必须要写,不写报java.lang.UnsatisfiedLinkError
    System.loadLibrary(Core.NATIVE_LIBRARY_NAME);

    File imgFile = new File("C:/Users/admin/Desktop/open/test.png");
    String dest = "C:/Users/admin/Desktop/open";
        
    //方式二
    Mat src = Imgcodecs.imread(imgFile.toString());
    Mat gray = new Mat();
    Imgproc.cvtColor(src, gray, Imgproc.COLOR_BGR2GRAY);
    src = gray;
    //保存灰度化的图片
    Imgcodecs.imwrite(dest + "/toGray2" + imgFile.getName(), src);
}

二值化:图像的二值化,就是将图像上的像素点的灰度值设置位0或255这两个极点,也就是将整个图像呈现出明显的只有黑和白的视觉效果

常见的二值化方法为固定阀值和自适应阀值,固定阀值就是制定一个固定的数值作为分界点,大于这个阀值的像素就设为255,小于该阀值就设为0,这种方法简单粗暴,但是效果不一定好.另外就是自适应阀值,每次根据图片的灰度情况找合适的阀值。自适应阀值的方法有很多,这里采用了一种类似K均值的方法,就是先选择一个值作为阀值,统计大于这个阀值的所有像素的灰度平均值和小于这个阀值的所有像素的灰度平均值,再求这两个值的平均值作为新的阀值。重复上面的计算,直到每次更新阀值后,大于该阀值和小于该阀值的像素数目不变为止。

代码如下

@Test
public void binaryzation(Mat mat) {
    int BLACK = 0;
    int WHITE = 255;
    int ucThre = 0, ucThre_new = 127;
    int nBack_count, nData_count;
    int nBack_sum, nData_sum;
    int nValue;
    int i, j;

    int width = mat.width(), height = mat.height();
    //寻找最佳的阙值
    while (ucThre != ucThre_new) {
        nBack_sum = nData_sum = 0;
        nBack_count = nData_count = 0;

        for (j = 0; j < height; ++j) {
            for (i = 0; i < width; i++) {
                nValue = (int) mat.get(j, i)[0];

                if (nValue > ucThre_new) {
                    nBack_sum += nValue;
                    nBack_count++;
                } else {
                    nData_sum += nValue;
                    nData_count++;
                }
            }
        }

        nBack_sum = nBack_sum / nBack_count;
        nData_sum = nData_sum / nData_count;
        ucThre = ucThre_new;
        ucThre_new = (nBack_sum + nData_sum) / 2;
    }
    
    //二值化处理
    int nBlack = 0;
    int nWhite = 0;
    for (j = 0; j < height; ++j) {
        for (i = 0; i < width; ++i) {
            nValue = (int) mat.get(j, i)[0];
            if (nValue > ucThre_new) {
                mat.put(j, i, WHITE);
                nWhite++;
            } else {
                mat.put(j, i, BLACK);
                nBlack++;
            }
        }
    }

    // 确保白底黑字
    if (nBlack > nWhite) {
        for (j = 0; j < height; ++j) {
            for (i = 0; i < width; ++i) {
                nValue = (int) (mat.get(j, i)[0]);
                if (nValue == 0) {
                    mat.put(j, i, WHITE);
                } else {
                    mat.put(j, i, BLACK);
                }
            }
        }
    }
}

测试二值化

@Test
public void binaryzation() {
    // 这个必须要写,不写报java.lang.UnsatisfiedLinkError
    System.loadLibrary(Core.NATIVE_LIBRARY_NAME);

    File imgFile = new File("C:/Users/admin/Desktop/open/test.png");
    String dest = "C:/Users/admin/Desktop/open";
    //先经过一步灰度化
    Mat src = Imgcodecs.imread(imgFile.toString());
    Mat gray = new Mat();
    Imgproc.cvtColor(src, gray, Imgproc.COLOR_BGR2GRAY);
    src = gray;
    //二值化
    binaryzation(src);
    Imgcodecs.imwrite(dest + "/binaryzation" + imgFile.getName(), src);
}

Opencv自己也提供了二值化的接口,好像没有上面的效果好,这里也把代码放出来

@Test

public  void  testOpencvBinary() {

 // 这个必须要写,不写报java.lang.UnsatisfiedLinkError

 System.loadLibrary(Core.NATIVE_LIBRARY_NAME);

 File imgFile = new  File("C:/Users/admin/Desktop/open/test.png");

 String dest = "C:/Users/admin/Desktop/open";

 Mat src = Imgcodecs.imread(imgFile.toString(), Imgcodecs.CV_LOAD_IMAGE_GRAYSCALE);

 Imgcodecs.imwrite(dest + "/AdaptiveThreshold1" + imgFile.getName(), src);

 Mat dst = new Mat();

 Imgproc.adaptiveThreshold(src, dst, 255, Imgproc.ADAPTIVE_THRESH_MEAN_C, Imgproc.THRESH_BINARY, 13, 5);

 Imgcodecs.imwrite(dest + "/AdaptiveThreshold2" + imgFile.getName(), dst);

 Imgproc.adaptiveThreshold(src, dst, 255, Imgproc.ADAPTIVE_THRESH_MEAN_C, Imgproc.THRESH_BINARY_INV, 13, 5);

 Imgcodecs.imwrite(dest + "/AdaptiveThreshold3" + imgFile.getName(), dst);

 Imgproc.adaptiveThreshold(src, dst, 255, Imgproc.ADAPTIVE_THRESH_GAUSSIAN_C, Imgproc.THRESH_BINARY, 13, 5);

 Imgcodecs.imwrite(dest + "/AdaptiveThreshold4" + imgFile.getName(), dst);

 Imgproc.adaptiveThreshold(src, dst, 255, Imgproc.ADAPTIVE_THRESH_GAUSSIAN_C, Imgproc.THRESH_BINARY_INV, 13, 5);

 Imgcodecs.imwrite(dest + "/AdaptiveThreshold5" + imgFile.getName(), dst);

}

本文章参考了很多博客,感谢;主要是跟着一个博客来实现的https://blog.csdn.net/ysc6688/article/category/2913009(也是基于opencv来做的,只不过他是用c++实现的)感谢

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容