javax.imageio.IIOException: Not a JPEG file: starts with 0x89 0x50

场景:在使用java对图片进行裁剪的情况下引发的错误。

异常日志如下:

javax.imageio.IIOException: Not a JPEG file: starts with 0x89 0x50
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.readImageHeader(Native Method)
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.readNativeHeader(JPEGImageReader.java:604)
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.checkTablesOnly(JPEGImageReader.java:342)
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.gotoImage(JPEGImageReader.java:476)
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.readHeader(JPEGImageReader.java:597)
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.readInternal(JPEGImageReader.java:1054)
    at com.sun.imageio.plugins.jpeg.JPEGImageReader.read(JPEGImageReader.java:1034)

很明显可以看出,报错原因是因为读取到的图片不是一个 JPEG 格式的图片。

裁剪图片的核心代码如下:

Iterator iterator = ImageIO.getImageReadersByFormatName("JPEG");   
ImageReader reader = (ImageReader)iterator.next();/*获取图片尺寸*/  
InputStream inputStream = new FileInputStream(sourcePath);
ImageInputStream iis = ImageIO.createImageInputStream(inputStream);
reader.setInput(iis, true);
ImageReadParam param = reader.getDefaultReadParam();
Rectangle rectangle = new Rectangle(0,0, destWidth, destHeight);        /*指定截取范围*/
param.setSourceRegion(rectangle);
BufferedImage bi = reader.read(0,param);

File outfile = new File(outputPath);
File dir = new File(outfile.getParent());
if(!dir.exists())  dir.mkdirs();

ImageIO.write(bi, "JPEG", outfile);     //JPEG、 PNG、 BMP

上面代码中读取和存储都用的是JPEG格式的文件方式?
出现这个错误的原因很简单:就是程序使用了JPEG格式的程序处理了非JPEG格式的图片,具体到上面的错误,是因为处理了PNG格式的图片导致的。

发生错误的代码是:

BufferedImage bi = reader.read(0,param);

但是,发生错误的实际原因是下面这一行读取图片内容:

//参数可以为 JPEG、 PNG、 BMP
Iterator iterator = ImageIO.getImageReadersByFormatName("JPEG");   

所以要处理起来也很简单,就是在裁剪程序中需要传入具体的格式,根据图片的实际格式进行处理,不能在程序中写死。
上面的方法可以有三种类型的格式:JPEG、 PNG、 BMP
具体获取格式的方法可以根据apache文件处理的类库

String extension = FilenameUtils.getExtension(filePath);

或者自行解析文件后缀。

但是,这并不是最完整的解决方案:
如果图片本身就是JEPG或者PNG的话都还好说,上面的方法能解决,但是如果图片本身是JEG格式的,人为修改后缀为PNG格式的,或者PNG图片通过修改后缀的方式改成了JPEG的图片,这样我们都知道在电脑上或者网页上是可以正常显示的,但是在程序中处理其实还是为报上面的错误。

所以,我们要通过另外的方法来判断图片的真实格式,通过读取图片的开头的几个字节来判断图片的真实格式,具体代码如下:

    public static String getFileType(String filePath) {
        FileInputStream fis = null;
        /**
         * 根据文件名称,获取后缀名的方式,但是不保险
         */
        String extension = FilenameUtils.getExtension(filePath);
        try {
            fis = new FileInputStream(new File(filePath));
            byte[] bs = new byte[1];
            fis.read(bs);
            String type = Integer.toHexString(bs[0]&0xFF);
            if("ff".equalsIgnoreCase(type))  extension = "JPEG";
            if("89".equalsIgnoreCase(type)) extension = "PNG";
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("获取图片类型出错 : " +  filePath);
        } finally {
            try{
                if(fis != null) fis.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return extension;
    }

通过上面方法获取到的图片类型算是比较靠谱的了。

将获取到的图片格式,传递到上面裁剪的方法中,替换里面写死的 “JPEG”参数,这样才能正确根据不同的图片进行处理。

各种图片不同的开始标识:

.jpg|jpeg   ff d8
.bmp    42 4d
.gif    47 49 46 38
.png    89 50 4e 47
.bz     42 5a
.zip    50 4b 03 04

就此解决上面问题:

参考文章:

https://www.acgist.com/article/134.html
http://www.2cto.com/kf/201207/140517.html

你可能感兴趣的:(java异常处理)