草庐IT

java - ImageIO 读取的 RGB 值与其他方法略有不同

coder 2023-05-16 原文

我发现我在使用 Java(实际上是 paint.NET)时获得的 RGB 与使用 ImageMagick、Gimp、Python 和 Octave 时不同。最后4个都同意,所以我假设是正确的。

对于这些示例,我使用的是以下测试图像:http://farm3.static.flickr.com/2811/9177301733_9836174725_o.jpg

测试像素x=4144 y=2768

               R    G    B
Java        = (125, 107, 69)
Paint.NET   = (125, 107, 69)
ImageMagick = (128, 106, 67)
Python      = (128, 106, 67)
Octave      = (128, 106, 67)
Gimp        = (128, 106, 67)

什么给了?

这是一个使用 imagemagick 的快速测试:

convert image.jpg -crop 1x1+4144+2768 -depth 8 txt:

输出:

# ImageMagick pixel enumeration: 1,1,65535,srgb
0,0: (32896,27242,17219)  #806A43  srgb(128,106,67)

这里有一些 java 和 python 代码也演示了这个问题:

import org.apache.commons.io.FileUtils;
import org.junit.Test;

import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.net.URISyntaxException;
import java.net.URL;

public class ImageIOTest {
    @Test
    public void can_read_file() throws IOException, InterruptedException, URISyntaxException {
        File tempFile = File.createTempFile("image", "jpg");
        FileUtils.copyURLToFile(new URL("http://farm3.static.flickr.com/2811/9177301733_9836174725_o.jpg"), tempFile);

        BufferedImage image = ImageIO.read(tempFile);

        int javaRGB = image.getRGB(4144, 2768);
        int javaRed = (javaRGB >> 16) & 0xFF;
        int javaGreen = (javaRGB >> 8) & 0xFF;
        int javaBlue = (javaRGB >> 0) & 0xFF;
        System.out.printf("rgb: (%d, %d, %d)", javaRed, javaGreen, javaBlue);
    }
}

这里是对应的python脚本:

from PIL import Image
import sys, urllib, cStringIO

file = cStringIO.StringIO(urllib.urlopen("http://farm3.static.flickr.com/2811/9177301733_9836174725_o.jpg").read())

im = Image.open(file)
pix = im.load()
print pix[4144, 2768]

我尝试过使用 this 12monkeys library希望那会解决它,但没有骰子。还有其他想法如何使用java提取正确的RGB值吗?当然我不是第一个遇到这个问题的人!

更新

我尝试了 getRaster().getSample() 但得到了同样的无效结果:System.out.println(raster.getSample(4144, 2768, 0)+","+ raster.getSample(4144, 2768, 1)+","+ raster.getSample(4144, 2768, 2)); 输出:125,107,69

更多信息

下面是一些输出,显示了图像左上角的前 9 个(3x3 平方)像素由三种不同工具解码的 RGB 值。如您所见,Python 和 ImageMagick 是一致的。 Java 有时会匹配。我在java不同意的地方放了一个X...:

Tool          [x, y] = (R , G , B )
ImageIO     : [0, 0] = (86, 90, 93)
Python      : [0, 0] = (86, 90, 93)
ImageMagick : [0, 0] = (86, 90, 93)

ImageIO     : [1, 0] = (86, 90, 93)
Python      : [1, 0] = (86, 90, 93)
ImageMagick : [1, 0] = (86, 90, 93)

ImageIO     : [2, 0] = (90, 91, 95) X
Python      : [2, 0] = (88, 92, 95)
ImageMagick : [2, 0] = (88, 92, 95)

ImageIO     : [0, 1] = (85, 93, 95)
Python      : [0, 1] = (85, 93, 95)
ImageMagick : [0, 1] = (85, 93, 95)

ImageIO     : [1, 1] = (85, 93, 95) X
Python      : [1, 1] = (87, 92, 95)
ImageMagick : [1, 1] = (87, 92, 95)

ImageIO     : [2, 1] = (87, 92, 95)
Python      : [2, 1] = (87, 92, 95)
ImageMagick : [2, 1] = (87, 92, 95)

ImageIO     : [0, 2] = (83, 93, 94)
Python      : [0, 2] = (83, 93, 94)
ImageMagick : [0, 2] = (83, 93, 94)

ImageIO     : [1, 2] = (83, 93, 94) X
Python      : [1, 2] = (84, 92, 94)
ImageMagick : [1, 2] = (84, 92, 94)

ImageIO     : [2, 2] = (83, 91, 93)
Python      : [2, 2] = (83, 91, 93)
ImageMagick : [2, 2] = (83, 91, 93)

为什么 Java 会为某些像素提供不同的值?或者,是否有另一种(快速)使用 native Java 代码生成正确值的方法?

2016 年 9 月 26 日更新:

我提交了演示此问题的代码并将其推送到 github (imageio-test),以便我可以轻松地在不同的机器上对其进行测试。事实证明,Java 在 OSX 和 Ubuntu Linux 上是一致的,但不一致的是 Python、ImageMagick 和 Octave。换句话说,在 Linux 机器上,所有工具都相互一致,因此,我现在认为 java 一直都是正确的,而在 OSX 上给出错误结果的是其他工具!我仍然不明白为什么,我也没有任何具体的证据证明哪些值是正确的,但我在某个地方......

最佳答案

实际上,我想扭转这个问题,并说我很惊讶这么多不同的平台和工具实际上产生了相同的值。 :-)

JPEG 有损

首先,JPEG 是一种有损图像压缩方法。这意味着复制原始的确切数据是不可能的。或者,如果您愿意,几个不同的像素值可能在某种程度上都是“正确的”。

并非所有 JPEG 软件都从同一源文件生成完全相同的值的技术原因通常是值的不同舍入/钳位,或浮点运算的整数近似值以获得更好的性能。其他变化可能源于用于恢复子采样色度值的不同插值算法,例如(即,更平滑的图像可能看起来更悦目,但不一定更正确)。

另一个 excellent answer对一个类似的问题指出“JPEG 标准不要求解码器实现产生逐位相同的输出图像”,并引用 the Wikipedia JPEG entry :

[...] precision requirements for the decoding [...]; the output from the reference algorithm must not exceed:

  • a maximum of one bit of difference for each pixel component
  • low mean square error over each 8×8-pixel block
  • very low mean error over each 8×8-pixel block
  • very low mean square error over the whole image
  • extremely low mean error over the whole image

(请注意,上面只讨论了引用实现)。

但是,幸运的是,您的所有软件/工具似乎最终都使用了(某些版本的)libjpeg。 .因为它们都使用 libjpeg,所以您看到的差异来源很可能与 JPEG 解码无关。

色彩空间

即使您的所有软件都将 JPEG 文件转换为使用 RGB 值的表示,它们用于此表示的颜色空间也可能存在差异。

看来您使用的所有软件实际上都在 sRGB color space 中显示 RGB 值。 .这可能是主流计算中使用的最标准和最广泛使用的色彩空间,所以这并不奇怪。由于色彩空间始终是 sRGB,因此您看到的差异来源很可能不是色彩空间。

ICC 配置文件和颜色匹配

颜色差异的下一个可能来源是颜色匹配(由颜色匹配模块、CMM 或颜色管理系统、CMS 完成)不是 100% 精确的科学(例如参见 this document on black point compensation 或阅读一些来自 Little CMS blog 的更多技术帖子)。

Mac OS X 上运行的软件很可能使用 Apple 的 CMM,而 Java 始终使用 Little CMS(来自 OpenJDK 7 或 Oracle JDK/JRE 8),Linux 平台上的大多数软件也可能使用开源Little CMS(根据 Little CMS 主页,“您可以在大多数 Linux 发行版中找到 Little CMS”)。 Windows 上的软件也可能会略有偏差(我无法验证 Paint.Net 是否使用 Little CMS、Windows 的内置 CMM 或其他东西)。当然,使用 Adob​​e 的 CMM(即 Photoshop)也可能会出现偏差。

同样,幸运的是,您测试的许多软件都使用相同的 CMM 或 CMS 引擎,Little CMS ,所以你会再次得到很多相同的结果。但您测试的某些软件似乎使用了不同的 CMM,并且可能是轻微色差的来源。

总结

您看到的不同像素值都是“正确的”。 差异源于软件中算法的不同实现或近似值,但这并不一定意味着一个值是正确的而其他值是正确的错了。

PS:如果您需要在多个平台上重现完全相同的值,请在所有平台上使用相同的工具堆栈/相同的算法。

关于java - ImageIO 读取的 RGB 值与其他方法略有不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39649292/

有关java - ImageIO 读取的 RGB 值与其他方法略有不同的更多相关文章

  1. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  2. Ruby 写入和读取对象到文件 - 2

    好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

  3. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  4. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  5. ruby - Rails 3 的 RGB 颜色选择器 - 2

    状态:我正在构建一个应用程序,其中需要一个可供用户选择颜色的字段,该字段将包含RGB颜色代码字符串。我已经测试了一个看起来很漂亮但效果不佳的。它是“挑剔的颜色”,并托管在此存储库中:https://github.com/Astorsoft/picky-color.在这里我打开一个关于它的一些问题的问题。问题:请建议我在Rails3应用程序中使用一些颜色选择器。 最佳答案 也许页面上的列表jQueryUIDevelopment:ColorPicker为您提供开箱即用的产品。原因是jQuery现在包含在Rails3应用程序中,因此使用基

  6. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

  7. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  8. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

  9. Observability:从零开始创建 Java 微服务并监控它 (二) - 2

    这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/

  10. 【Java 面试合集】HashMap中为什么引入红黑树,而不是AVL树呢 - 2

    HashMap中为什么引入红黑树,而不是AVL树呢1.概述开始学习这个知识点之前我们需要知道,在JDK1.8以及之前,针对HashMap有什么不同。JDK1.7的时候,HashMap的底层实现是数组+链表JDK1.8的时候,HashMap的底层实现是数组+链表+红黑树我们要思考一个问题,为什么要从链表转为红黑树呢。首先先让我们了解下链表有什么不好???2.链表上述的截图其实就是链表的结构,我们来看下链表的增删改查的时间复杂度增:因为链表不是线性结构,所以每次添加的时候,只需要移动一个节点,所以可以理解为复杂度是N(1)删:算法时间复杂度跟增保持一致查:既然是非线性结构,所以查询某一个节点的时候

随机推荐