关闭 x
IT技术网
    技 采 号
    ITJS.cn - 技术改变世界
    • 实用工具
    • 菜鸟教程
    IT采购网 中国存储网 科技号 CIO智库

    IT技术网

    IT采购网
    • 首页
    • 行业资讯
    • 系统运维
      • 操作系统
        • Windows
        • Linux
        • Mac OS
      • 数据库
        • MySQL
        • Oracle
        • SQL Server
      • 网站建设
    • 人工智能
    • 半导体芯片
    • 笔记本电脑
    • 智能手机
    • 智能汽车
    • 编程语言
    IT技术网 - ITJS.CN
    首页 » JAVA »Java图像灰度化的实现过程解析

    Java图像灰度化的实现过程解析

    2015-03-26 00:00:00 出处:偶my耶的博客
    分享

    概要

    ITJS的这篇文章主要介绍了灰度化的几种方法,以及如何使用Java实现灰度化。同时分析了网上一种常见却并不妥当的Java灰度化实现,以及证明了opencv的灰度化是使用“加权灰度化”法

    24位彩色图与8位灰度图

    首先要先介绍一下24位彩色图像,在一个24位彩色图像中,每个像素由三个字节表示,通常表示为RGB。通常,许多24位彩色图像存储为32位图像,每个像素多余的字节存储为一个alpha值,表现有特殊影响的信息[1]。

    在RGB模型中,如果R=G=B时,则彩色表示一种灰度颜色,其中R=G=B的值叫灰度值,因此,灰度图像每个像素只需一个字节存放灰度值(又称强度值、亮度值),灰度范围为0-255[2]。这样就得到一幅图片的灰度图。

    几种灰度化的方法

    分量法:使用RGB三个分量中的一个作为灰度图的灰度值。 最值法:使用RGB三个分量中最大值或最小值作为灰度图的灰度值。 均值法:使用RGB三个分量的平均值作为灰度图的灰度值。 加权法:由于人眼颜色敏感度不同,按下一定的权值对RGB三分量进行加权平均能得到较合理的灰度图像。一般情况按照:Y = 0.30R + 0.59G + 0.11B。

    [注]加权法实际上是取一幅图片的亮度值作为灰度值来计算,用到了YUV模型。在[3]中会发现作者使用了Y = 0.21 * r + 0.71 * g + 0.07 * b来计算灰度值(显然三个权值相加并不等于1,可能是作者的错误?)。实际上,这种差别应该与是否使用伽马校正有关[1]。

    一种Java实现灰度化的方法

    如果你搜索“Java实现灰度化”,十有八九都是一种方法(代码):

    public void grayImage() throws IOException{
        File file = new File(System.getProperty("user.dir")+"/test.jpg");
        BufferedImage image = ImageIO.read(file);
    
        int width = image.getWidth();  
        int height = image.getHeight();  
    
        BufferedImage grayImage = new BufferedImage(width, height, BufferedImage.TYPE_BYTE_GRAY); 
        for(int i= 0 ; i < width ; i++){  
            for(int j = 0 ; j < height; j++){  
            int rgb = image.getRGB(i, j);  
            grayImage.setRGB(i, j, rgb);  
            }  
        }  
    
        File newFile = new File(System.getProperty("user.dir")+"/method1.jpg");  
        ImageIO.write(grayImage, "jpg", newFile);  
    }

    test.jpg的原图为:

    使用上述方法得到的灰度图:

    看到这幅灰度图,似乎还真是可行,但是如果我们使用opencv来实现灰度化或使用PIL(Python),你会发现效果相差很大:

    img = cv2.imread('test.jpg',cv2.IMREAD_COLOR)
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    cv2.imwrite('PythonMethod.jpg', gray)

    可以清楚的看到,使用opencv(PIL也是一样的)得到的灰度图要比上面Java方法得到的方法好很多,很多细节都能够看得到。这说明,网上这种流行的方法一直都存在这某种问题,只是一直被忽略。

    opencv如何实现灰度化

    如果读过opencv相关的书籍或代码,大概都能知道opencv灰度化使用的是加权法,之所以说是大概,因为我们不知道为什么opencv灰度化的图像如此的好,是否有其他的处理细节被我们忽略了?

    验证我们的猜想很简单,只要查看像素值灰度化前后的变化就知道了,可以如下测试:

    img = cv2.imread('test.jpg',cv2.IMREAD_COLOR)
    h, w = img.shape[:2]
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    for j in range(w):
    	for i in range(h):
    		print str(i) + " : " + str(j) + " " + str(gray[i][j])
    print img[h-1][w-1][0:3]

    以下打印了这么多像素点,我们也很难判断,但是我们只要关注一下最后一个像素点,就能够发现端倪: 原图最后的像素点RGB值为44,67,89,而灰度化之后的值为71。正好符合加权法计算的灰度值。如果你检查之前用Java灰度化的图片的像素值,你会发现不仅仅像素值不符合这个公式,甚至相差甚远。

    到此,我们猜测opencv(也包括PIL)是使用加权法实现的灰度化。

    Java实现加权法灰度化

    如果网上那段流行的方法不行,我们该如何使用Java实现灰度化?实际上[3]已经成功的实现了(多种方法的)灰度化(外国友人搞技术还是很给力的),在此仅仅提取必要的代码:

    private static int colorToRGB(int alpha, int red, int green, int blue) {
    
    		int newPixel = 0;
    		newPixel += alpha;
    		newPixel = newPixel << 8;
    		newPixel += red;
    		newPixel = newPixel << 8;
    		newPixel += green;
    		newPixel = newPixel << 8;
    		newPixel += blue;
    
    		return newPixel;
    
    }
    public static void main(String[] args) throws IOException {
    	BufferedImage bufferedImage 
    		= ImageIO.read(new File(System.getProperty("user.dir" + "/test.jpg"));
    	BufferedImage grayImage = 
    		new BufferedImage(bufferedImage.getWidth(), 
    						  bufferedImage.getHeight(), 
    						  bufferedImage.getType());
    
    	for (int i = 0; i < bufferedImage.getWidth(); i++) {
    		for (int j = 0; j < bufferedImage.getHeight(); j++) {
    			final int color = bufferedImage.getRGB(i, j);
    			final int r = (color >> 16) & 0xff;
    			final int g = (color >> 8) & 0xff;
    			final int b = color & 0xff;
    			int gray = (int) (0.3 * r + 0.59 * g + 0.11 * b);;
    			System.out.println(i + " : " + j + " " + gray);
    			int newPixel = colorToRGB(255, gray, gray, gray);
    			grayImage.setRGB(i, j, newPixel);
    		}
    	}
    	File newFile = new File(System.getProperty("user.dir") + "/ok.jpg");
    	ImageIO.write(grayImage, "jpg", newFile);
    }

    上面的代码会打印出灰度化后的像素值,如果再与上面的Python代码做对比,你会发现像素值完全的对应上了。colorToRGB方法中对彩色图的处理正好是4个字节,其中之一是alpha参数(前文所讲),下图是这段代码灰度化后的图像:

    对于其他方法,依次同理可得。

    总结

    ITJS的这篇文章的成因本是希望使用Java实现几种灰度化操作,并使用opencv来验证转化的对错,但在实际测试中发现了一些问题(转化后的图片有差异,以及如何在转化后根据灰度值生成灰度图等问题),并就此进行了一定的思考与验证。

    这里需要注意的是,网上的一些文章或多或少没有做更进一步的思考(甚至很多都是照搬,尤其是国内的文章),而对于这些实际问题,动手实现并验证是非常重要的方法。希望ITJS的这篇文章对大家有所帮助。

    参考

    [1] 《多媒体技术教程》Ze-Nian Li,Mark S.Drew著,机械工业出版社。 [2] 百度百科:灰度值 [3] Java color image to grayscale conversion algorithm(s)
    上一篇返回首页 下一篇

    声明: 此文观点不代表本站立场;转载务必保留本文链接;版权疑问请联系我们。

    别人在看

    正版 Windows 11产品密钥怎么查找/查看?

    还有3个月,微软将停止 Windows 10 的更新

    Windows 10 终止支持后,企业为何要立即升级?

    Windows 10 将于 2025年10 月终止技术支持,建议迁移到 Windows 11

    Windows 12 发布推迟,微软正全力筹备Windows 11 25H2更新

    Linux 退出 mail的命令是什么

    Linux 提醒 No space left on device,但我的空间看起来还有不少空余呢

    hiberfil.sys文件可以删除吗?了解该文件并手把手教你删除C盘的hiberfil.sys文件

    Window 10和 Windows 11哪个好?答案是:看你自己的需求

    盗版软件成公司里的“隐形炸弹”?老板们的“法务噩梦” 有救了!

    IT头条

    公安部:我国在售汽车搭载的“智驾”系统都不具备“自动驾驶”功能

    02:03

    液冷服务器概念股走强,博汇、润泽等液冷概念股票大涨

    01:17

    亚太地区的 AI 驱动型医疗保健:2025 年及以后的下一步是什么?

    16:30

    智能手机市场风云:iPhone领跑销量榜,华为缺席引争议

    15:43

    大数据算法和“老师傅”经验叠加 智慧化收储粮食尽显“科技范”

    15:17

    技术热点

    商业智能成CIO优先关注点 技术落地方显成效(1)

    用linux安装MySQL时产生问题破解

    JAVA中关于Map的九大问题

    windows 7旗舰版无法使用远程登录如何开启telnet服务

    Android View 事件分发机制详解

    MySQL用户变量的用法

      友情链接:
    • IT采购网
    • 科技号
    • 中国存储网
    • 存储网
    • 半导体联盟
    • 医疗软件网
    • 软件中国
    • ITbrand
    • 采购中国
    • CIO智库
    • 考研题库
    • 法务网
    • AI工具网
    • 电子芯片网
    • 安全库
    • 隐私保护
    • 版权申明
    • 联系我们
    IT技术网 版权所有 © 2020-2025,京ICP备14047533号-20,Power by OK设计网

    在上方输入关键词后,回车键 开始搜索。Esc键 取消该搜索窗口。