【问题标题】:Getting Pixel Values from Byte Array从字节数组中获取像素值
【发布时间】:2016-11-15 22:09:54
【问题描述】:

我无法获取像素数据。

我的程序截取屏幕截图,每个循环它都会存储之前的屏幕截图。

我的目标是在当前屏幕截图和旧屏幕截图之间进行像素级别的比较。

我运行了这段代码,它告诉我屏幕截图的格式: System.out.println(image.getType());

这个(对于我的程序)的输出是 1 表示它是 BufferedImage.TYPE_INT_RGB

根据我读取的内容,类型决定了像素值在字节数组中的顺序。

我正在使用此代码将我的缓冲图像转换为字节数组(缓冲图像是使用 awt.Robot 类创建的):

public byte[] convertToByteArray(BufferedImage img){
        byte[] imageInByte = null;
        try {



            // convert BufferedImage to byte array
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            ImageIO.write(img, "png", baos);
            baos.flush();
            imageInByte = baos.toByteArray();
            baos.close();

        } catch (IOException ex) {
            Logger.getLogger(OverlayWindow.class.getName()).log(Level.SEVERE, null, ex);
        }
        return imageInByte;
}

最后我使用比较方法来检查字节数组。现在只打印数组的颜色值:

  final byte[] pixels = convertToByteArray(image);
  final int pixelLength = 3;
        for (int pixel = 0, row = 0, col = 0; pixel < 1; pixel += pixelLength) {
        int argb = 0;
        argb += -16777216; // 255 alpha
        argb += ((int) pixels[pixel] & 0xff); // blue
        argb += (((int) pixels[pixel + 1] & 0xff) << 8); // green
        argb += (((int) pixels[pixel + 2] & 0xff) << 16); // red


        int r = (argb >> 16) & 0xff, g = (argb >> 8)& 0xff, b = argb & 0xff;
        System.out.println("R = " + r);
        System.out.println("G = " + g);
        System.out.println("B = " +  b);

        col++;
        if (col == width) {
           col = 0;
           row++;
        }




     }

我对这段代码的问题是,即使我截取了纯色的屏幕截图,像素值也到处都是。我希望每个像素都具有相同的颜色值。

-编辑-

出于性能原因,我避免使用 getRGB。在我的应用程序中,每次调用 getRGB 遍历两个大图像非常昂贵。

【问题讨论】:

  • 我们不知道会发生什么 - 你有一个完整的运行程序,包含你所做的所有细节吗??
  • 首先,pixels 不是真正的像素值,它是压缩的 PNG 文件的字节......所以无论如何这都行不通。其次,如果您已经将像素值作为单独的 R、G 和 B 样本(您的代码假定),为什么要尝试将它们打包成 32 位 ARGB 格式并再次解包?我建议首先使用int argb = image.getRGB(x, y);,然后像你已经做的那样分成R、G和B。然后从那里优化,如果它不够快......
  • 我更新了我的问题。那么字节数组有问题吗?我最初尝试使用它来获取字节数组:((DataBufferByte) bufferedImage.getData().getDataBuffer()).getData(); 当我尝试运行它时会引发以下错误:java.lang.ClassCastException: java.awt.image.DataBufferInt cannot be cast to java.awt.image.DataBufferByte 那么获取字节数组以进行像素比较的正确方法是什么?

标签: java arrays image-processing bufferedimage


【解决方案1】:

访问 BufferedImage 中像素值的最简单方法是使用 Raster:

BufferedImage image = ...
for (int y=0 ; y < image.getHeight() ; y++)
    for (int x=0 ; x < image.getWidth() ; x++)
        for (int c=0 ; c < image.getRaster().getNumBands() ; c++)
            final int value = image.getRaster().getSample(x, y, c) ; // Returns the value of the channel C of the pixel (x,y)

光栅将为您处理编码,使其成为访问像素值的最简单方法。然而,最快的方法是使用 DataBuffer,但是你必须管理所有的编码。

/* This method takes a BufferedImage encoded with TYPE_INT_ARGB and copies the pixel values into an image encoded with TYPE_4BYTE_ABGR.*/
public static void IntToByte(BufferedImage source, BufferedImage result)
    {
    final byte[] bb = ((DataBufferByte)result.getRaster().getDataBuffer()).getData() ;
    final int[] ib  = ((DataBufferInt)source.getRaster().getDataBuffer()).getData() ;

    switch ( source.getType() )
        {
        case BufferedImage.TYPE_INT_ARGB :
            for (int i=0, b=0 ; i < ib.length ; i++, b+=4)
                {
                int p   = ib[i] ;
                bb[b]   = (byte)((p & 0xFF000000) >> 24) ;
                bb[b+3] = (byte)((p & 0xFF0000) >> 16) ;
                bb[b+2] = (byte)((p & 0xFF00) >> 8) ;
                bb[b+1] = (byte)( p & 0xFF) ;
                }
            break ;
        // Many other case to manage...
        }
    }

【讨论】:

  • BufferedImage.TYPE_INT_RGBBufferedImage.TYPE_INT_ARGB 的格式类型相同吗?我也不需要将它编码成另一个图像,但我可以看到你在哪里展示如何获取像素值,我唯一的问题是 ((DataBufferByte)result.getRaster().getDataBuffer()).getData() 抛出异常:java.lang.ClassCastException: java.awt.image.DataBufferInt cannot be cast to java.awt.image.DataBufferByte
  • BufferedImage.TYPE_INT_RGB 和 BufferedImage.TYPE_INT_ARGB 是同一个 int 编码,只是 Alpha 通道存在与否。
  • 例外是因为您想将 DataBufferInt 转换为 DataBufferByte。因此,您尝试通过将其转换为字节来访问使用 int 编码的图像。正如我所说,使用 DataBuffer 时,您必须管理所有编码。
  • hmm 好的,所以我尝试使用 DataBufferInt 而不是 Byte,这不会引发异常,但我不确定如何操作这个数组,有什么可以参考我的,以帮助我从中获取像素数据这个?
  • 是的,我的函数 IntToByte,我读取的是 DataBufferInt 的源图像,并提取所有像素的值。
猜你喜欢
  • 1970-01-01
  • 2019-07-23
  • 2016-05-18
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2016-09-20
  • 2021-03-29
  • 1970-01-01
相关资源
最近更新 更多