无法使用Samsung手机创建的java加载JPEG

Dav*_*vid 14 java jpeg javafx

我在使用javafx加载由三星Galaxy S7边缘拍摄的JPEG图像时遇到问题(图片可在https://www.dropbox.com/s/w6lvdnqwcgw321s/20171122_140732.jpg?dl=0获得).我正在使用Image类来加载图像.

import java.io.FileInputStream;

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.image.Image;
import javafx.scene.image.ImageView;
import javafx.scene.layout.Pane;
import javafx.stage.Stage;

public class JPEGProblem extends Application {

   public static void main(String[] args) {
      launch(args);
   }

   @Override
   public void start(Stage window) throws Exception {
      Image img = new Image(new FileInputStream("/path/to/image.jpg"));
      if (img.getException() != null)
         throw img.getException();

      ImageView imgView = new ImageView(img);
      window.setScene(new Scene(new Pane(imgView)));
      window.show();
   }

}
Run Code Online (Sandbox Code Playgroud)

试图加载图像的构造函数调用在错误流上打印以下错误消息:

2018年2月4日晚上11:48:23 com.sun.javafx.tk.quantum.PrismImageLoader2 $ PrismLoadListener imageLoadWarning警告:顺序JPEG的SOS参数无效

我从image对象获得的异常是一个IOException,其中包含以下消息:

不支持的标记类型0x65

我做了一些研究,事实证明,这是三星手机拍摄的全景图像的一个已知问题.正如在这个帖子中指出的那样:https://forums.adobe.com/thread/2131432,一些0xFF字节,表示后面的字节是元信息而不是实际数据,不会通过添加以下0x00字节来转义0xFF.
但是我试着编写操作图像数据的代码,以便添加丢失的0x00字节,但结果比预期的要复杂得多,我不想编写自己的JPEG解析器/加载器.
有些程序可以显示那些无效的JPEG图像,例如Microsoft Fotos或Paint.看起来他们容忍这些无效图像,将这些虚假标记视为内容数据.
有没有办法用java加载这些无效的图像,而不用自己处理单个字节?

Eni*_*igo 11

你的问题太棒了,真让我思考并搜索了几个小时)

这是一个有点hacky解决方案(没有第三方库),我最终得到:

import javafx.application.Application;
import javafx.embed.swing.SwingFXUtils;
import javafx.scene.Scene;
import javafx.scene.image.ImageView;
import javafx.scene.layout.Pane;
import javafx.stage.Stage;

import java.awt.*;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.DataBuffer;
import java.awt.image.DataBufferInt;
import java.awt.image.DirectColorModel;
import java.awt.image.PixelGrabber;
import java.awt.image.Raster;
import java.awt.image.WritableRaster;

public class JPEGProblem extends Application {

    public static void main(String[] args) {
        launch(args);
    }

    @Override
    public void start(Stage window) throws Exception {
        BufferedImage bi = getBufferedImage();
        ImageView imgView = getFinalImageView(bi);
        window.setScene(new Scene(new Pane(imgView)));
        window.show();
    }

    private BufferedImage getBufferedImage() throws InterruptedException {
        final java.awt.Image image = Toolkit.getDefaultToolkit().createImage("path\to\file");

        final int[] RGB_MASKS = {0xFF0000, 0xFF00, 0xFF};
        final ColorModel RGB_OPAQUE =
                new DirectColorModel(32, RGB_MASKS[0], RGB_MASKS[1], RGB_MASKS[2]);

        PixelGrabber pg = new PixelGrabber(image, 0, 0, -1, -1, true);
        pg.grabPixels();
        int width = pg.getWidth(), height = pg.getHeight();
        DataBuffer buffer = new DataBufferInt((int[]) pg.getPixels(), pg.getWidth() * pg.getHeight());
        WritableRaster raster = Raster.createPackedRaster(buffer, width, height, width, RGB_MASKS, null);
        return new BufferedImage(RGB_OPAQUE, raster, false, null);
    }

    private ImageView getFinalImageView(BufferedImage bi) throws Exception {
        ImageView imgView = new ImageView(SwingFXUtils.toFXImage(bi, null));
        imgView.setFitWidth(1024);
        imgView.setFitHeight(756);
        imgView.setRotate(180);
        return imgView;
    }
}
Run Code Online (Sandbox Code Playgroud)

这里的问题是标准的Imageapi无法读取"损坏的"图像,因此我们需要以不同的方式阅读它.对于此Toolkit.getDefaultToolkit(),可以使用createImage()方法.实际上,这getBufferedImage部分是从这个答案中取出来的,所以,所有的学分都去了那里)

getFinalImageView方法中,我们只需将其BufferedImage转换为javafx Image然后ImageView使用ImageIO类.

结果: 在此输入图像描述

注意! 我仍然可以在日志中观察到一些异常,但它们不会阻止此代码成功执行.