
在使用java的`imageio.read()`方法加载数字相机拍摄的直立图片时,常会遇到`bufferedimage`的宽度和高度与实际显示不符的问题。这通常是由于jpeg图像中的exif元数据未被正确解析所致。本文将深入探讨此问题根源,并提供两种基于第三方库(如thumbnailator和twelvemonkeys)的解决方案,以确保图像尺寸的准确获取和处理。
理解问题根源:Exif元数据与图像方向
当通过ImageIO.read(stream)加载图片并尝试获取其宽度和高度时,例如:
File imgFile = new File("path/to/your/image.jpg");
FileImageInputStream stream = new FileImageInputStream(imgFile);
BufferedImage srcImage = ImageIO.read(stream);
int actualHeight = srcImage.getHeight();
int actualWidth = srcImage.getWidth();对于横向(landscape)图片,actualWidth和actualHeight通常会返回正确的值。然而,对于数码相机或手机拍摄的直立(portrait)图片,我们可能会发现actualWidth和actualHeight的值是颠倒的,例如,一张原始尺寸为200x500(宽x高)的图片,可能会得到actualHeight = 200和actualWidth = 500的结果。
这种现象的根本原因在于,大多数数码相机传感器在物理上以固定的方向(通常是横向)记录像素数据。当用户拍摄直立照片时,图像的实际显示方向并非通过旋转像素数据来改变,而是通过在图像文件中嵌入Exif(Exchangeable image file format)元数据中的“方向”标签来指示。标准的ImageIO JPEG插件在读取图像时,通常只处理像素数据,而不会解析和应用Exif方向标签,因此返回的是像素数据存储的原始方向尺寸,而非用户期望的显示方向尺寸。
解决方案:处理Exif方向信息
为了正确获取直立图片的宽度和高度,我们需要一个能够解析Exif方向元数据并据此调整图像的库。以下介绍两种常用的Java库及其使用方法。
立即学习“Java免费学习笔记(深入)”;
方案一:使用Thumbnailator库
Thumbnailator是一个功能强大的Java图像处理库,它能够自动处理Exif方向信息,并在加载图像时进行相应的旋转。
-
添加Maven依赖:
net.coobird thumbnailator 0.4.19 -
加载图像并获取尺寸: 使用Thumbnailator加载图像时,它会自动处理Exif方向。
import net.coobird.thumbnailator.Thumbnails; import java.awt.image.BufferedImage; import java.io.File; import java.io.FileInputStream; import java.io.IOException; public class ImageOrientationFixer { public static void main(String[] args) { File imgFile = new File("path/to/your/portrait_image.jpg"); try (FileInputStream fis = new FileInputStream(imgFile)) { // 使用Thumbnailator加载图片,scale(1)表示不缩放 BufferedImage srcImage = Thumbnails.of(fis) .scale(1) .asBufferedImage(); int actualWidth = srcImage.getWidth(); int actualHeight = srcImage.getHeight(); System.out.println("Thumbnailator处理后的图片宽度: " + actualWidth); System.out.println("Thumbnailator处理后的图片高度: " + actualHeight); } catch (IOException e) { e.printStackTrace(); } } }通过这种方式,srcImage将是一个已经根据Exif方向信息旋转过的BufferedImage,其getWidth()和getHeight()方法会返回正确的显示尺寸。
方案二:使用TwelveMonkeys ImageIO插件
TwelveMonkeys是一个为Java ImageIO框架提供扩展的库,它包含了对多种图像格式和元数据的支持,包括Exif方向。它的EXIFUtilities类可以直接读取带有正确方向的图像。
-
添加Maven依赖: TwelveMonkeys通常需要多个模块,其中imageio-jpeg和imageio-metadata是处理Exif的关键。
com.twelvemonkeys.imageio imageio-jpeg 3.10.0 com.twelvemonkeys.imageio imageio-metadata 3.10.0 请注意,为了让TwelveMonkeys插件生效,通常需要在JVM启动时或代码中确保其被正确注册。在大多数情况下,只要存在于classpath中,ImageIO会自动发现并使用这些插件。
-
加载图像并获取尺寸:EXIFUtilities.readWithOrientation()方法会返回一个IIOImage对象,其中包含已根据Exif方向调整的RenderedImage。
import com.twelvemonkeys.imageio.metadata.exif.EXIFUtilities; import javax.imageio.IIOImage; import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException; public class ImageOrientationFixerTwelveMonkeys { public static void main(String[] args) { File imgFile = new File("path/to/your/portrait_image.jpg"); try { // 使用EXIFUtilities读取图片,自动处理方向 IIOImage image = EXIFUtilities.readWithOrientation(imgFile); BufferedImage srcImage = (BufferedImage) image.getRenderedImage(); int actualWidth = srcImage.getWidth(); int actualHeight = srcImage.getHeight(); System.out.println("TwelveMonkeys处理后的图片宽度: " + actualWidth); System.out.println("TwelveMonkeys处理后的图片高度: " + actualHeight); } catch (IOException e) { e.printStackTrace(); } } }如果不需要访问其他Exif元数据,可以更简洁地获取BufferedImage:
BufferedImage srcImage = (BufferedImage) EXIFUtilities.readWithOrientation(imgFile) .getRenderedImage();
注意事项与总结
- 相机设置: 某些数码相机提供“机内旋转”选项,可以将直立图片直接旋转为正确的像素方向并保存,从而避免Exif方向问题。但此功能通常默认关闭,且依赖于对输入图像源的控制,在通用场景下不总是可行。
- 性能考量: 图像旋转和Exif解析会增加一定的处理开销。对于需要处理大量图像的场景,建议进行性能测试并选择最适合的方案。
- 库选择: Thumbnailator更侧重于图像缩放和操作,而TwelveMonkeys则是一个更全面的ImageIO插件集合,提供了对多种格式和元数据的支持。根据项目需求选择合适的库。
- 兼容性: 确保所选库的版本与您的Java环境兼容。
综上所述,当在Java中处理来自数码设备(尤其是手机)的图像时,务必考虑到Exif方向元数据可能带来的宽高颠倒问题。通过集成Thumbnailator或TwelveMonkeys这类专门处理Exif的库,可以有效地解决这一问题,确保图像尺寸的准确获取和后续处理的正确性。










