从 Eclipse 和命令行运行时,BufferedImage 字节具有不同的字节顺序

2024-03-20

我试图转换一个BufferedImage's byte[]从 32 位 RGBA 到 24 位 RGB。根据这个答案 https://stackoverflow.com/a/9470843/2581401最快的方式获得byte[]从图像中可以看出:

byte[] pixels = ((DataBufferByte) bufferedImage.getRaster().getDataBuffer()).getData();

因此,我迭代所有字节,假设它们的顺序为 R G B A,并且对于每 4 个字节,我将前 3 个写入输出 byte[](即忽略 alpha 值)。

当从 Eclipse 运行并且字节转换正确时,此方法工作正常。但是,当我从命令行运行相同的程序时,会以相反的字节顺序返回相同的字节!

The test image I use for my test is a 5x5 black image where only its top-left corner is different having the RGBA color [aa cc ee ff]: Test image 5x5

为了方便起见,还提供了放大版本:

我的文件夹结构是:

- src/  
  - test.png
  - test/
      - TestBufferedImage.java  

SSCCE 的内容如下:

package test;

import java.awt.image.BufferedImage;
import java.awt.image.DataBufferByte;
import java.io.IOException;
import java.io.InputStream;

import javax.imageio.ImageIO;

public class TestBufferedImage {

  private static void log(String s) {
    System.out.println(s);
  }

  private static String toByteString(byte b) {
    // Perform a bitwise AND for convenience while printing. 
    // Otherwise Integer.toHexString() interprets values as integers and a negative byte 0xFF will be printed as "ffffffff"
    return Integer.toHexString(b & 0xFF);
  }

  /**
   * @param args
   * @throws IOException 
   */
  public static void main(String[] args) throws IOException {

    InputStream stream = TestBufferedImage.class.getClassLoader().getResourceAsStream("test.png");
    BufferedImage image = ImageIO.read(stream);
    stream.close();
    log("Image loaded succesfully, width=" + image.getWidth() + " height=" + image.getHeight());

    log("Converting from 32-bit to 24-bit...");
    DataBufferByte buffer = (DataBufferByte)image.getRaster().getDataBuffer(); 
    byte[] input = buffer.getData();
    byte[] output = convertTo24Bit(input);
    log("Converted total of " + input.length + " bytes to " + output.length + " bytes");
  }

  private static byte[] convertTo24Bit(byte[] input) {
    int dataLength = input.length;
    byte[] convertedData = new byte[ dataLength * 3 / 4 ];

    for (int i = 0, j = 0; i < dataLength; i+=4, j+=3) {
      convertIntByteToByte(input, i, convertedData, j);
    }
    return convertedData;
  }

  private static void convertIntByteToByte(byte[] src, int srcIndex, byte[] out, int outIndex) {
    byte r = src[srcIndex];
    byte g = src[srcIndex+1];
    byte b = src[srcIndex+2];
    byte a = src[srcIndex+3];

    out[outIndex] = r;
    out[outIndex+1] = g; 
    out[outIndex+2] = b; 

    log("i=" + srcIndex 
        + " Converting [" + toByteString(r) + ", " + toByteString(g) 
        + ", " + toByteString(b) + ", " + toByteString(a) + "] --> ["
        + toByteString(out[outIndex]) + ", " + toByteString(out[outIndex+1])
        + ", " + toByteString(out[outIndex+2]) + "]"
        );
  }

}

从 Eclipse 运行时的输出(版本:Juno Service Release 2 Build id:20130225-0426):

Image loaded succesfully, width=5 height=5
Converting from 32-bit to 24-bit...
i=0 Converting [aa, cc, ee, ff] --> [aa, cc, ee]   // <-- Bytes have the correct order
i=4 Converting [0, 0, 0, ff] --> [0, 0, 0]
i=8 Converting [0, 0, 0, ff] --> [0, 0, 0]
.....
i=96 Converting [0, 0, 0, ff] --> [0, 0, 0]
Converted total of 100 bytes to 75 bytes

从命令行 (Windows Vista) 运行时的输出java test.TestBufferedImage:

Image loaded succesfully, width=5 height=5
Converting from 32-bit to 24-bit...
i=0 Converting [ff, ee, cc, aa] --> [ff, ee, cc]    // <-- Bytes are returned with a different byte order!
i=4 Converting [ff, 0, 0, 0] --> [ff, 0, 0]
i=8 Converting [ff, 0, 0, 0] --> [ff, 0, 0]
.....
i=96 Converting [ff, 0, 0, 0] --> [ff, 0, 0]
Converted total of 100 bytes to 75 bytes

那么有没有人遇到过类似的问题和/或可以解释到底发生了什么?为什么从 Eclipse 内部运行时字节顺序不同?


在回答我自己的问题之前,我真的、真的、真的感谢@Joni 和@haraldK,他们为我指明了正确的方向。我对内部结构的了解BufferedImage, ColorModel, SampleModel类似的事情不太好,所以他们帮助了我。

所以这就是发生的事情:

首先,不同的行为是由不同的 JRE 引起的。打印java版本的日志语句显示Eclipse打印1.6.0_16-b01当命令行打印时1.6.0_31-b05。显然是图像加载的实现(这将是PNGImageReader类)在版本之间发生了变化,我怀疑它在这个变化 https://bugs.java.com/bugdatabase/view_bug?bug_id=6549882.

尽管两个版本使用相同的ColorModel and SampleModel所以我无法理解这个变化(对我来说这似乎是一个真正的密码破译者)所以我进一步调查了。

两个版本之间的重要变化PNGImageReader曾在Iterator<ImageTypeSpecifier> getImageTypes()决定可用于创建新图像的可用兼容格式的方法:

版本1.6.0_16-b01:

...
case PNG_COLOR_RGB_ALPHA:
    // Component R, G, B, A (non-premultiplied)
    rgb = ColorSpace.getInstance(ColorSpace.CS_sRGB);
    bandOffsets = new int[4];
    bandOffsets[0] = 0;
    bandOffsets[1] = 1;
    bandOffsets[2] = 2;
    bandOffsets[3] = 3;

    l.add(ImageTypeSpecifier.createInterleaved(rgb,
                                               bandOffsets,
                                               dataType,
                                               true,
                                               false));
    break;

版本1.6.0_31-b05:

...
case PNG_COLOR_RGB_ALPHA:
    if (bitDepth == 8) {
        // some standard types of buffered images
        // wich can be used as destination
        l.add(ImageTypeSpecifier.createFromBufferedImageType(
                                           BufferedImage.TYPE_4BYTE_ABGR));

        l.add(ImageTypeSpecifier.createFromBufferedImageType(
                                           BufferedImage.TYPE_INT_ARGB));
    }

    // Component R, G, B, A (non-premultiplied)
    rgb = ColorSpace.getInstance(ColorSpace.CS_sRGB);
    bandOffsets = new int[4];
    bandOffsets[0] = 0;
    bandOffsets[1] = 1;
    bandOffsets[2] = 2;
    bandOffsets[3] = 3;

    l.add(ImageTypeSpecifier.createInterleaved(rgb,
                                               bandOffsets,
                                               dataType,
                                               true,
                                               false));
    break;

所以在新版本中,新ImageTypeSpecifier告诉 PNG 加载器,内部表示的兼容图像是BufferedImage.TYPE_4BYTE_ABGR由于它是列表中的第一种类型,因此加载程序会继续使用它。这就是颜色通道的带(以及字节)被反转的原因。

意识到上述情况后,我突然意识到这并不是我想象的错误或代码破坏者。原因是因为我正在使用byte[] pixels = ((DataBufferByte) bufferedImage.getRaster().getDataBuffer()).getData();为了获取字节,我实际上是在侵入内部数据结构(即忽略SampleModel)的图像。合同中没有任何内容ImageReader保证字节的顺序。如果需要,它可以自由地更改其内部数据结构(这就是封装的要点,对吧?)。正确配置的方法ImageReader如果您想要除默认行为之外的任何其他内容,请获取其默认行为ImageReadParam并配置它。然后使用将其传回给读者reader.read(imageIndex, param);

因为我实际上希望读者返回图像字节的特定格式,所以我应该这样做:

log("Java version: " + System.getProperty("java.runtime.version"));

// get the reader
ImageReader ir = ImageIO.getImageReadersByFormatName("png").next();

// get the default param    
ImageReadParam p = ir.getDefaultReadParam();
p.setDestinationType(
    // define the image type to return if supported
    ImageTypeSpecifier.createInterleaved(
        ColorSpace.getInstance(ColorSpace.CS_sRGB), 
        new int[] {0, 1, 2, 3},    // <-- the order of the color bands to return so the bytes are in the desired order
        DataBuffer.TYPE_BYTE,
        true, false)
        );

InputStream stream = TestBufferedImage.class.getClassLoader().getResourceAsStream("test.png");

ImageInputStream imageStream = ImageIO.createImageInputStream(stream);
ir.setInput(imageStream);
BufferedImage image = ir.read(0, p); 

现在,两个版本都将以相同的 RGBA 形式返回字节顺序,即在两种情况下都会打印不同颜色的输出:

...
i=0 Converting [aa, cc, ee, ff] --> [aa, cc, ee] 
...
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

从 Eclipse 和命令行运行时,BufferedImage 字节具有不同的字节顺序 的相关文章

  • 如何使用 Java 和 Selenium WebDriver 在 C 目录中创建文件夹并需要将屏幕截图保存在该目录中?

    目前正在与硒网络驱动程序和代码Java 我有一种情况 我需要在 C 目录中创建一个文件夹 并在该文件夹中创建我通过 selenium Web 驱动程序代码拍摄的屏幕截图 它需要存储在带有时间戳的文件夹中 如果我每天按计划运行脚本 所有屏幕截
  • 为什么 i++ 不是原子的?

    Why is i Java 中不是原子的 为了更深入地了解 Java 我尝试计算线程中循环的执行频率 所以我用了一个 private static int total 0 在主课中 我有两个线程 主题 1 打印System out prin
  • Play框架运行应用程序问题

    每当我尝试运行使用以下命令创建的新 Web 应用程序时 我都会收到以下错误Play http www playframework org Error occurred during initialization of VM Could no
  • 给定两个 SSH2 密钥,我如何检查它们是否属于 Java 中的同一密钥对?

    我正在尝试找到一种方法来验证两个 SSH2 密钥 一个私有密钥和一个公共密钥 是否属于同一密钥对 我用过JSch http www jcraft com jsch 用于加载和解析私钥 更新 可以显示如何从私钥 SSH2 RSA 重新生成公钥
  • 反射找不到对象子类型

    我试图通过使用反射来获取包中的所有类 当我使用具体类的代码 本例中为 A 时 它可以工作并打印子类信息 B 扩展 A 因此它打印 B 信息 但是当我将它与对象类一起使用时 它不起作用 我该如何修复它 这段代码的工作原理 Reflection
  • JavaMail 只获取新邮件

    我想知道是否有一种方法可以在javamail中只获取新消息 例如 在初始加载时 获取收件箱中的所有消息并存储它们 然后 每当应用程序再次加载时 仅获取新消息 而不是再次重新加载它们 javamail 可以做到这一点吗 它是如何工作的 一些背
  • Liferay ClassNotFoundException:DLFileEntryImpl

    在我的 6 1 0 Portal 实例上 带有使用 ServiceBuilder 和 DL Api 的 6 1 0 SDK Portlet 这一行 DynamicQuery query DynamicQueryFactoryUtil for
  • 斯坦福 NLP - 处理文件列表时 OpenIE 内存不足

    我正在尝试使用斯坦福 CoreNLP 中的 OpenIE 工具从多个文件中提取信息 当多个文件 而不是一个 传递到输入时 它会给出内存不足错误 All files have been queued awaiting termination
  • 如何为俚语和表情符号构建正则表达式 (regex)

    我需要构建一个正则表达式来匹配俚语 即 lol lmao imo 等 和表情符号 即 P 等 我按照以下示例进行操作http www coderanch com t 497238 java java Regular Expression D
  • Java按日期升序对列表对象进行排序[重复]

    这个问题在这里已经有答案了 我想按一个参数对对象列表进行排序 其日期格式为 YYYY MM DD HH mm 按升序排列 我找不到正确的解决方案 在 python 中使用 lambda 很容易对其进行排序 但在 Java 中我遇到了问题 f
  • Android Studio 0.4.3 Eclipse项目没有gradle

    在此版本之前 在 Android Studio 中按原样打开 Eclipse 项目似乎很容易 无需任何转换 我更喜欢 Android Studio 环境 但我正在开发一个使用 eclipse 作为主要 IDE 的项目 我不想只为这个项目下载
  • JRE 系统库 [WebSphere v6.1 JRE](未绑定)

    将项目导入 Eclipse 后 我的构建路径中出现以下错误 JRE System Library WebSphere v6 1 JRE unbound 谁知道怎么修它 右键单击项目 特性 gt Java 构建路径 gt 图书馆 gt JRE
  • Java执行器服务线程池[关闭]

    很难说出这里问的是什么 这个问题是含糊的 模糊的 不完整的 过于宽泛的或修辞性的 无法以目前的形式得到合理的回答 如需帮助澄清此问题以便重新打开 访问帮助中心 help reopen questions 如果我使用 Executor 框架在
  • Google App Engine 如何预编译 Java?

    App Engine 对应用程序的 Java 字节码使用 预编译 过程 以增强应用程序在 Java 运行时环境中的性能 预编译代码的功能与原始字节码相同 有没有详细的信息这是做什么的 我在一个中找到了这个谷歌群组消息 http groups
  • 如何在桌面浏览器上使用 webdriver 移动网络

    我正在使用 selenium webdriver 进行 AUT 被测应用程序 的功能测试自动化 AUT 是响应式网络 我几乎完成了桌面浏览器的不同测试用例 现在 相同的测试用例也适用于移动浏览器 因为可以从移动浏览器访问 AUT 由于它是响
  • 在activity_main.xml中注释

    我是安卓新手 据我所知 XML 中的注释与 HTML 中的注释相同 使用 形式 我想在 Android 项目的 Activity main xml 配置文件中写一些注释 但它给了我错误 值得注意的是 我使用的是 Eclipse 但目前 我直
  • 在 Maven 依赖项中指定 jar 和 test-jar 类型

    我有一个名为 commons 的项目 其中包含运行时和测试的常见内容 在主项目中 我添加了公共资源的依赖项
  • 捕获的图像分辨率太大

    我在做什么 我允许用户捕获图像 将其存储到 SD 卡中并上传到服务器 但捕获图像的分辨率为宽度 4608 像素和高度 2592 像素 现在我想要什么 如何在不影响质量的情况下获得小分辨率图像 例如我可以获取或设置捕获的图像分辨率为原始图像分
  • java.lang.IllegalStateException:驱动程序可执行文件的路径必须由 webdriver.chrome.driver 系统属性设置 - Similiar 不回答

    尝试学习 Selenium 我打开了类似的问题 但似乎没有任何帮助 我的代码 package seleniumPractice import org openqa selenium WebDriver import org openqa s
  • 将 List 转换为 JSON

    Hi guys 有人可以帮助我 如何将我的 HQL 查询结果转换为带有对象列表的 JSON 并通过休息服务获取它 这是我的服务方法 它返回查询结果列表 Override public List

随机推荐

  • 如何获取多字节字符串的字节大小

    如何在 Visual C 中获取多字节字符串的字节大小 有没有函数或者我必须自己计算字符 或者 更一般地说 如何获得 TCHAR 字符串的正确字节大小 解决方案 tcslen T TCHAR string sizeof TCHAR EDIT
  • 这个吐司是从哪里来的?

    由于某种原因 我正在开发的应用程序正在显示一个 toast 显示我的设备上剩余的内部存储空间 即使我没有对此进行编码 这是一个屏幕截图 https i stack imgur com z2ERU png https i stack imgu
  • Python 2.7 的蓝牙? [关闭]

    Closed 这个问题正在寻求书籍 工具 软件库等的推荐 不满足堆栈溢出指南 help closed questions 目前不接受答案 对 Python 2 7 进行蓝牙编程的最佳方法是什么 我尝试使用pybluez https code
  • 使用预定脚本部署 Shiny App

    我有一个简单的脚本 可以在 Rstudio 中用于部署应用程序 rsconnect setAccountInfo name xx token xx secret xx xx library rsconnect deployApp xxx l
  • 链接后台 NSURLSession 上传

    有人成功链接 NSURLSession 后台上传吗 我正在尝试使用 NSURLSession 的后台上传来上传 5 MB 的巨大视频文件 上传必须按顺序进行 整个事情在前台工作得很好 我为此使用 AFNetwoking 它是多部分上传 但是
  • 尝试从一个页面推送到另一页面时出现“找不到组件工厂”错误

    尝试从一个页面推送到另一页面时出现错误 当我尝试推送到同一页面时 它不会给出该错误 只有我在从一个页面推送到另一页面时遇到错误 setRoot 也没有给出错误 this navCtrl push Page7 我已将 Page7 添加到 ap
  • Pandas fillna() 基于特定列属性

    假设我有这张桌子 Type Killed Survived Dog 5 2 Dog 3 4 Cat 1 7 Dog nan 3 cow nan 2 其中的价值之一Killed缺少 Type Dog 我想将平均值归咎于 Killed for
  • 最好的 SQL Server 性能优化技术是什么? [关闭]

    Closed 这个问题需要多问focused help closed questions 目前不接受答案 我始终采取的方法是首先使用最少的索引集部署数据库 然后根据性能要求添加 更改索引 这种方法效果相当好 但是 它仍然没有告诉我可以在哪里
  • Rails 4.0 的自定义错误处理

    我正在使用 Ruby 2 0 和 Rails 4 0 构建 Ruby on Rails api 我的应用程序几乎只是一个 JSON API 因此如果发生错误 500 404 我想捕获该错误并返回格式良好的 JSON 错误消息 我试过了thi
  • 子集化 data.frame 时的 NA 会发生一些意想不到的事情

    考虑以下代码 当你没有明确测试NA在您的情况下 该代码将在稍后您的数据发生更改时失败 gt A toy example gt a lt as data frame cbind col1 c 1 2 3 4 col2 c 2 NA 2 3 c
  • 如何使 MapView 对象透明(alpha)?

    All 我需要在 MapView 对象上显示信息 那里没有问题 问题是 有时 MapView 对象显示的地图详细信息在视觉上与我的叠加数据相竞争 因此 我想做的是提供一种通过使用 alpha 通道在视觉上 缩小 MapView 对象的方法
  • 如何自动检测用户的时区?

    您好 我正在创建一个网络应用程序 如果用户注册 我们将显示创建日期 为此 我们在我的sql表中使用当前时间戳 它显示服务器时间 但我们不知道如何根据用户时区转换时间 因为我们不是获取用户所在国家 地区 任何人都可以帮我解决它吗 提前致谢 使
  • “return 0”和“exit (0)”之间的区别[重复]

    这个问题在这里已经有答案了 有什么区别吗return 0 and exit 0 在函数中使用时 如果是 我应该什么时候使用return 0 or exit 0 在一个函数中 return退出该函数 同时exit退出程序 In main函数执
  • backbone.js 收集事件

    我开发了一个 jquery 和backbone js 网络应用程序 一个组件有一个 html 表 该表后面是一个backbone js 集合 该集合中的任何更改都会导致 html 表的更新 所以我写 this collection bind
  • UITableView重新加载数据

    我正在为 iPhone 制作一个基于导航的应用程序 我的视图控制器之一如下所示 interface NewComputerViewController UIViewController
  • 什么时候会使用 BRICK 权限?

    在Android中 曾经有一个名为BRICK http developer android com reference android Manifest permission html BRICK可用于潜在地禁用该设备 除了将其视为都市神话
  • 除 None 之外的任何类型的 Mypy 注释[重复]

    这个问题在这里已经有答案了 我怎样才能注释一个类型 除了None 换句话说 这个类型是Any但不是None 你可以做Union int str 但排除None来自那个工会
  • Scala Futures 如何与 flatMap 链接在一起?

    我正在 Scala 中首次使用 Futures 并且正在研究使用 flatMap 组合器的示例 我一直在关注这个讨论 http docs scala lang org overviews core futures html http doc
  • 当构建系统已引用 System.Core 时,添加对 System.Core 的引用

    即使项目已生成 Visual Studio Intellisense 也无法识别动态关键字 我尝试添加对System Core来解决问题 我收到此错误 无法添加对 System Core 的引用 该组件是 已经被构建系统自动引用 我注意到我
  • 从 Eclipse 和命令行运行时,BufferedImage 字节具有不同的字节顺序

    我试图转换一个BufferedImage s byte 从 32 位 RGBA 到 24 位 RGB 根据这个答案 https stackoverflow com a 9470843 2581401最快的方式获得byte 从图像中可以看出