场景:在使用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