Java一分钟之-图像处理:BufferedImage与ImageIO

在Java中,处理图像通常涉及到BufferedImage类和ImageIO工具类。本文将简要介绍这两个关键组件,以及在处理图像时常见的问题、易错点及如何避免它们。
在这里插入图片描述

1. BufferedImage

BufferedImage是Java中的一个图像类,它提供了对图像像素级别的访问和修改。你可以创建一个新的BufferedImage,或者从文件或流中加载现有图像:

BufferedImage image = ImageIO.read(new File("image.jpg"));

2. ImageIO

ImageIO是Java的标准图像输入输出工具类,用于读取、写入和处理各种格式的图像文件。例如,你可以使用ImageIO.write()方法将BufferedImage保存到文件:

ImageIO.write(image, "JPEG", new File("output.jpg"));

常见问题与易错点

3.1 图像格式不支持

ImageIO.read()ImageIO.write()方法依赖于可用的插件来处理特定的图像格式。如果没有安装相应的插件,读取或写入某些格式的图像会抛出IOException

3.2 图像尺寸处理不当

在缩放或裁剪图像时,如果没有正确处理图像的宽度和高度,可能会导致图像变形或丢失部分像素。

BufferedImage scaledImage = new BufferedImage(newWidth, newHeight, image.getType());
Graphics2D g2d = scaledImage.createGraphics();
g2d.drawImage(image, 0, 0, newWidth, newHeight, null);
g2d.dispose();

3.3 忽略异常处理

ImageIO的读写方法可能会抛出异常,如IOException。忽略这些异常可能导致程序崩溃,而不提供任何反馈。

3.4 未正确关闭流

如果从流中读取图像,记得在操作完成后关闭流,否则可能导致资源泄漏。

try (InputStream in = new FileInputStream("image.jpg")) {
    BufferedImage image = ImageIO.read(in);
} catch (IOException e) {
    e.printStackTrace();
}

如何避免这些问题

  • 检查插件支持:确保你的Java环境支持所需的图像格式,或者使用第三方库如Apache Commons Imaging。
  • 正确处理图像尺寸:在缩放或裁剪时,保持图像的宽高比,或使用适当的算法防止失真。
  • 捕获并处理异常:在调用ImageIO方法时,务必包含异常处理代码,以便在出现问题时能够优雅地处理。
  • 关闭流:使用try-with-resources语句或其他方式确保流在使用后被正确关闭。

通过了解这些常见问题和解决方案,你将能更有效地在Java中处理图像。在实际项目中,记得不断实践和优化你的代码,以提高性能和稳定性。祝你在图像处理的道路上越走越远!

public void start(Stage primaryStage) throws Exception { String CONNECTION_URL = "jdbc:hive2://192.168.88.100:10000/default"; String USERNAME = "root"; String PASSWORD = "123456"; String JDBC_DRIVER = "org.apache.hive.jdbc.HiveDriver"; // 加载 Hive JDBC 驱动 Class.forName(JDBC_DRIVER); // 建立 JDBC 连接 Connection conn = DriverManager.getConnection(CONNECTION_URL, USERNAME, PASSWORD); // 创建 Statement 对象 Statement stmt = conn.createStatement(); // 执行查询语句 String sql = "SELECT data FROM picture_01 WHERE dt = '2023-07-13'"; ResultSet rs = stmt.executeQuery(sql); // 处理查询结果 while (rs.next()) { // 处理数据,例如打印输出 System.out.println(rs.getBinaryStream("data")); } // 从 Hive 数据库获取二进制数据的输入流对象 InputStream inputStream = rs.getBinaryStream("data"); // 将输入流对象转换为 BufferedImage 对象 BufferedImage bufferedImage = ImageIO.read(inputStream); // 转换 BufferedImageJavaFX 的 Image 对象 Image image = SwingFXUtils.toFXImage(bufferedImage, null); System.out.println("1"); // 创建图像视图 ImageView imageView = new ImageView(image); // 创建场景,并将图像视图添加到场景中 Scene scene = new Scene(imageView.getParent()); // 设置舞台并显示场景 primaryStage.setTitle("Image Viewer"); primaryStage.setScene(scene); primaryStage.show(); // 关闭连接和释放资源 rs.close(); stmt.close(); conn.close(); // 关闭输入流 inputStream.close(); } }此代码报错上述代码报错java.lang.reflect.InvocationTargetException at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:498) at com.sun.javafx.application.LauncherImpl.launchApplicationWithArgs(LauncherImpl.java:389)什么原因
07-17
根据你提供的代码和错误信息,这个问题可能是由于在 JavaFX 的 `start` 方法中直接执行了数据库查询和 UI 操作,导致在 JavaFX 应用程序线程中发生了阻塞操作,从而引发了 `InvocationTargetException` 异常。 在 JavaFX 应用程序中,`start` 方法是在主线程上调用的,用于初始化和显示应用程序的 UI。然而,数据库查询是一个耗时操作,应该在后台线程中执行,而不是直接在 `start` 方法中执行。 为了解决这个问题,可以将数据库查询的代码放在后台线程中执行,并在查询完成后更新 UI。可以使用 `javafx.concurrent.Task` 或 `javafx.concurrent.Service` 类来实现这个目的。 以下是一个使用 `javafx.concurrent.Task` 的示例代码: ```java public void start(Stage primaryStage) throws Exception { String CONNECTION_URL = "jdbc:hive2://192.168.88.100:10000/default"; String USERNAME = "root"; String PASSWORD = "123456"; String JDBC_DRIVER = "org.apache.hive.jdbc.HiveDriver"; Task<BufferedImage> queryTask = new Task<BufferedImage>() { @Override protected BufferedImage call() throws Exception { Class.forName(JDBC_DRIVER); Connection conn = DriverManager.getConnection(CONNECTION_URL, USERNAME, PASSWORD); Statement stmt = conn.createStatement(); String sql = "SELECT data FROM picture_01 WHERE dt = '2023-07-13'"; ResultSet rs = stmt.executeQuery(sql); rs.next(); InputStream inputStream = rs.getBinaryStream("data"); BufferedImage bufferedImage = ImageIO.read(inputStream); rs.close(); stmt.close(); conn.close(); inputStream.close(); return bufferedImage; } }; queryTask.setOnSucceeded(event -> { BufferedImage bufferedImage = queryTask.getValue(); Image image = SwingFXUtils.toFXImage(bufferedImage, null); ImageView imageView = new ImageView(image); Scene scene = new Scene(imageView.getParent()); primaryStage.setTitle("Image Viewer"); primaryStage.setScene(scene); primaryStage.show(); }); queryTask.setOnFailed(event -> { Throwable exception = queryTask.getException(); // 处理异常 exception.printStackTrace(); }); Thread queryThread = new Thread(queryTask); queryThread.start(); } ``` 这样,数据库查询将在后台线程中执行,完成后将触发 `onSucceeded` 事件,其中可以更新 UI。如果查询过程中发生异常,将触发 `onFailed` 事件,在该事件中可以处理异常情况。 请注意,这只是一个示例代码,你需要根据实际情况进行调整。希望能帮助你解决问题!如果有任何问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Jimaks

您的鼓励将是我创作的最大动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值