今天使用ImageIO实现裁切图片,遇到了一个错误:
Not a JPEG file: starts with 0xff 0xd9
因为要获取裁剪原点和裁切尺寸,所以我想获取图片宽高,用较小的一个作为裁切尺寸,进而得到一个正方形的图,所以就有了如下代码:
File imageFile = new File(imageFileFullPath);
FileInputStream imageFileInputStream = new FileInputStream(imageFile);
// 获取原始图片,用于读取宽和高
BufferedImage bufferedImage = ImageIO.read(imageFileInputStream);
int imageWidth = bufferedImage.getWidth();
int imageHeight = bufferedImage.getHeight();
if(imageWidth == imageHeight) {
//直接返回,不进行裁切
}
// 取较小的数值为裁剪尺寸
int clipSize = imageWidth > imageHeight ? imageHeight : imageWidth;
然而在裁切并输出结果时,又用到了这样的代码:
Iterator<ImageReader> readerIt = ImageIO.getImageReadersByFormatName("jpg");
ImageReader reader = readerIt.next();
// 获取图片流
ImageInputStream imageInputStream = ImageIO.createImageInputStream(imageFileInputStream);
reader.setInput(imageInputStream, true);
Rectangle rectangle = new Rectangle(0, 0, clipSize, clipSize);
param.setSourceRegion(rectangle);
//读取选取的区域
BufferedImage clipedImage = reader.read(0, param);
没过多久我就找到了解决办法,以至于我认为导致错误的原因就是定义了两个BufferedImage。这个解决方法是:去掉获取宽高的BufferedImage对象的定义,改用reader获取原始图片尺寸:
int imageWidth = reader.getWidth(0);
int imageHeight = reader.getHeight(0);
这样就能够正常运行了。以目前掌握的知识还不能理解文章开头提到的错误是如何出现的,因为去掉一个BufferedImage的定义就能够正常裁切,所以我想是不是不能定义两个BufferedImage对象的原因,直到我尝试了以下方法:
BufferedImage clipedImage = bufferedImage.getSubimage(0, 0, clipSize, clipSize);
取代上面的那一大段代码,同样用到了两个BufferedImage对象,这种形式就能够很好地去执行。
之后我又继续思考可能导致这个错误的原因,发现了一个可疑之处:
// 获取原始图片,用于读取宽和高
BufferedImage bufferedImage = ImageIO.read(imageFileInputStream);
这个imageFileInputStream对象是已经读取到的图片对象,而ImageIO的read方法还有一个参数就是File,所以我替换了一下:
BufferedImage bufferedImage = ImageIO.read(imageFile);
竟然也裁切成功了!不过还是没有从根本上理解错误出现的原因,暂且记下,以后慢慢发现。
图片裁切的代码参考了:
http://www.oschina.net/code/snippet_857368_14826
再贴一个BufferedImage相关的好文:
http://blog.163.com/yuhua_kui/blog/static/9679964420139118203818/