【Java进阶】图像处理:从基础概念掌握实际操作
一、核心概念:BufferedImage - 图像的画布与数据载体
在Java图像处理的世界里,BufferedImage
是当之无愧的核心。你可以将它想象成一块内存中的画布,所有的像素数据、颜色模型以及图像的宽度、高度等信息都存储在其中。
BufferedImage
继承自Image
类,但它提供了更丰富的操作,比如直接访问像素、获取图像的颜色模型等。当你从文件读取一张图片时,通常会将其加载为BufferedImage
对象。
为什么是BufferedImage
?
- 内存驻留: 图像数据直接存储在内存中,方便快速读写和操作。
- 像素级访问: 提供了
getRGB(x, y)
和setRGB(x, y, rgb)
等方法,允许你直接操作每个像素的颜色。 - 丰富的构造器: 支持多种颜色模型(如
TYPE_INT_RGB
,TYPE_INT_ARGB
等)和数据类型,满足不同需求。
二、图像的读写:ImageIO - Java与图像文件的桥梁
Java的javax.imageio.ImageIO
类是处理图像文件输入/输出的利器。它支持多种常见的图像格式,如JPEG、PNG、GIF、BMP等。
1. 读取图像
从文件或输入流中加载图像非常简单:
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;public class ImageReadWrite {public static void main(String[] args) {// 假设你有一个名为 "input.jpg" 的图片文件File inputFile = new File("input.jpg");BufferedImage originalImage = null;try {originalImage = ImageIO.read(inputFile);System.out.println("图片读取成功!宽度: " + originalImage.getWidth() + ", 高度: " + originalImage.getHeight());} catch (IOException e) {System.err.println("读取图片失败: " + e.getMessage());}// 接下来可以对 originalImage 进行操作...}
}
2. 写入图像
将BufferedImage
对象保存为图片文件同样简单:
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;public class ImageReadWrite {public static void main(String[] args) {// 假设 originalImage 是你已经处理过的 BufferedImage 对象BufferedImage processedImage = new BufferedImage(100, 100, BufferedImage.TYPE_INT_RGB); // 示例:创建一个空白图片File outputFile = new File("output.png"); // 指定输出文件名和格式try {// 参数1: 要写入的BufferedImage对象// 参数2: 图像格式 (如 "png", "jpg", "gif")// 参数3: 输出文件对象ImageIO.write(processedImage, "png", outputFile);System.out.println("图片写入成功!保存为: " + outputFile.getAbsolutePath());} catch (IOException e) {System.err.println("写入图片失败: " + e.getMessage());}}
}
小贴士: ImageIO.write()
的第二个参数指定了图像的格式。Java会根据这个字符串选择合适的写入器。如果你想查看系统支持的所有格式,可以使用ImageIO.getReaderFormatNames()
和ImageIO.getWriterFormatNames()
。
三、基本图像操作
掌握了BufferedImage
的读写,我们就可以开始进行一些基本的图像操作了!
1. 图像缩放 (Resizing)
图像缩放是图像处理中最常见的操作之一。Java提供了多种方式实现,其中使用Graphics2D
是更推荐的做法,因为它能提供更好的缩放质量。
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.RenderingHints;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;public class ImageOperations {public static BufferedImage resizeImage(BufferedImage originalImage, int targetWidth, int targetHeight) {// 创建一个新的BufferedImage对象,用于存放缩放后的图像BufferedImage resizedImage = new BufferedImage(targetWidth, targetHeight, originalImage.getType());// 获取Graphics2D对象,用于绘制Graphics2D g2d = resizedImage.createGraphics();// 开启高质量的渲染提示,如抗锯齿、双线性插值等g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);// 将原图像绘制到新的BufferedImage上,自动进行缩放g2d.drawImage(originalImage, 0, 0, targetWidth, targetHeight, null);g2d.dispose(); // 释放资源return resizedImage;}public static void main(String[] args) throws IOException {BufferedImage original = ImageIO.read(new File("input.jpg"));BufferedImage resized = resizeImage(original, 200, 150); // 缩放到200x150ImageIO.write(resized, "jpg", new File("output_resized.jpg"));System.out.println("图片缩放完成!");}
}
2. 图像裁剪 (Cropping)
裁剪图像通常涉及到获取BufferedImage
的一个子区域。BufferedImage
的getSubimage()
方法可以轻松实现这一点。
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;public class ImageOperations {public static BufferedImage cropImage(BufferedImage originalImage, int x, int y, int width, int height) {// 检查裁剪区域是否有效if (x < 0 || y < 0 || x + width > originalImage.getWidth() || y + height > originalImage.getHeight()) {throw new IllegalArgumentException("裁剪区域超出图像边界!");}// 使用getSubimage方法获取子图像return originalImage.getSubimage(x, y, width, height);}public static void main(String[] args) throws IOException {BufferedImage original = ImageIO.read(new File("input.jpg"));// 裁剪图像:从(50, 50)点开始,裁剪一个100x80的区域BufferedImage cropped = cropImage(original, 50, 50, 100, 80);ImageIO.write(cropped, "jpg", new File("output_cropped.jpg"));System.out.println("图片裁剪完成!");}
}
3. 像素级操作:黑白滤镜 (Grayscale)
BufferedImage
允许我们直接访问并修改每个像素的颜色。下面我们来实现一个简单的黑白滤镜:
import java.awt.Color;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;public class ImageOperations {public static BufferedImage toGrayscale(BufferedImage originalImage) {// 创建一个新的BufferedImage,类型为灰度(如果有Alpha通道,也可以是TYPE_INT_ARGB)BufferedImage grayscaleImage = new BufferedImage(originalImage.getWidth(), originalImage.getHeight(), originalImage.getType());for (int y = 0; y < originalImage.getHeight(); y++) {for (int x = 0; x < originalImage.getWidth(); x++) {int rgb = originalImage.getRGB(x, y); // 获取像素的RGB值 (int类型)Color color = new Color(rgb, true); // 将int值转换为Color对象,true表示包含Alpha通道// 获取R, G, B分量int red = color.getRed();int green = color.getGreen();int blue = color.getBlue();int alpha = color.getAlpha();// 计算灰度值(常见的加权平均法)int gray = (int) (0.299 * red + 0.587 * green + 0.114 * blue);// 创建新的灰度颜色Color grayColor = new Color(gray, gray, gray, alpha);grayscaleImage.setRGB(x, y, grayColor.getRGB()); // 设置新的像素值}}return grayscaleImage;}public static void main(String[] args) throws IOException {BufferedImage original = ImageIO.read(new File("input.jpg"));BufferedImage grayscale = toGrayscale(original);ImageIO.write(grayscale, "jpg", new File("output_grayscale.jpg"));System.out.println("图片转为灰度完成!");}
}
四、进阶与优化:突破边界
1. 颜色模型与性能
BufferedImage
支持多种图像类型(TYPE_INT_RGB
, TYPE_INT_ARGB
, TYPE_BYTE_BINARY
等),选择合适的类型可以优化内存使用和处理性能。例如,如果你的图像不需要透明度,使用TYPE_INT_RGB
会比TYPE_INT_ARGB
更高效。
对于大量像素操作,直接操作BufferedImage
的Raster
数据(像素数组)通常比getRGB
/setRGB
方法更快,因为后者会进行额外的类型转换。
2. AffineTransformOp - 图像变换的利器
对于旋转、剪切、翻转等几何变换,java.awt.image.AffineTransformOp
提供了更专业和高效的解决方案。它基于java.awt.geom.AffineTransform
来定义变换矩阵。
import java.awt.geom.AffineTransform;
import java.awt.image.AffineTransformOp;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;public class ImageTransform {public static BufferedImage rotateImage(BufferedImage originalImage, double angleDegrees) {double angleRadians = Math.toRadians(angleDegrees);// 计算旋转后的图像尺寸double sin = Math.abs(Math.sin(angleRadians));double cos = Math.abs(Math.cos(angleRadians));int w = originalImage.getWidth();int h = originalImage.getHeight();int newWidth = (int) Math.floor(w * cos + h * sin);int newHeight = (int) Math.floor(h * cos + w * sin);// 创建旋转变换AffineTransform transform = new AffineTransform();// 移动到中心点,然后旋转,再移动回中心点(保证在图像中心旋转)transform.translate(newWidth / 2, newHeight / 2);transform.rotate(angleRadians);transform.translate(-originalImage.getWidth() / 2, -originalImage.getHeight() / 2);// 创建操作对象,指定渲染质量AffineTransformOp op = new AffineTransformOp(transform, AffineTransformOp.TYPE_BILINEAR);// 创建一个新的BufferedImage来存放旋转后的图像BufferedImage rotatedImage = new BufferedImage(newWidth, newHeight, originalImage.getType());// 执行变换return op.filter(originalImage, rotatedImage);}public static void main(String[] args) throws IOException {BufferedImage original = ImageIO.read(new File("input.jpg"));BufferedImage rotated = rotateImage(original, 45); // 旋转45度ImageIO.write(rotated, "png", new File("output_rotated.png"));System.out.println("图片旋转完成!");}
}
3. 外部库的助攻
虽然Java内置的API功能强大,但对于更复杂的图像处理任务(如特征识别、机器学习、更专业的滤镜效果),或者追求极致性能,你可能需要借助一些成熟的外部库:
- OpenCV (JavaCV): 计算机视觉领域的巨头,提供C++原生库的Java封装,性能卓越,功能极其丰富(人脸识别、物体检测、图像分割等)。
- ImageJ: 一个强大的开源图像处理平台,主要用于科学图像分析,提供了大量的算法和插件。
- Thumbnailator: 一个专注于创建缩略图和水印的轻量级库,API设计简洁直观,适合快速实现常见需求。
- MarvinFramework: 另一个纯Java的图像处理框架,提供了丰富的图像滤镜、边缘检测、图像分割等功能。
这些库通常提供了比AWT/Swing更高效的实现和更高级的算法,能够大大简化开发复杂图像应用的工作。
相关文章:
【Java进阶】图像处理:从基础概念掌握实际操作
一、核心概念:BufferedImage - 图像的画布与数据载体 在Java图像处理的世界里,BufferedImage是当之无愧的核心。你可以将它想象成一块内存中的画布,所有的像素数据、颜色模型以及图像的宽度、高度等信息都存储在其中。 BufferedImage继承自…...

JAVA网络编程——socket套接字的介绍下(详细)
目录 前言 1.TCP 套接字编程 与 UDP 数据报套接字的区别 2.TCP流套接字编程 API 介绍 TCP回显式服务器 Scanner 的多种使用方式 PrintWriter 的多种使用方式 TCP客户端 3. TCP 服务器中引入多线程 结尾 前言 各位读者大家好,今天笔者继续更新socket套接字的下半部分…...
Apache SeaTunnel 引擎深度解析:原理、技术与高效实践
Apache SeaTunnel 作为新一代高性能分布式数据集成平台,其核心引擎设计融合了现代大数据处理架构的精髓。 Apache SeaTunnel引擎通过分布式架构革新、精细化资源控制及企业级可靠性设计,显著提升了数据集成管道的执行效率与运维体验。其模块化设计允许用…...
深入理解 Maven 循环依赖问题及其解决方案
在 Java 开发领域,Maven 作为主流构建工具极大简化了依赖管理和项目构建。然而**循环依赖(circular dependency)**问题仍是常见挑战,轻则导致构建失败,重则引发类加载异常和系统架构混乱。 本文将从根源分析循环依赖的…...
pytest中的元类思想与实战应用
在Python编程世界里,元类是一种强大而高级的特性,它能在类定义阶段深度定制类的创建与行为。而pytest作为热门的测试框架,虽然没有直接使用元类,但在设计机制上,却暗含了许多与元类思想相通的地方。接下来,…...
前端生成UUID
UUID(Universally Unique Identifier)是一种在分布式系统中广泛使用的标识符,具有全球唯一性。在前端开发中,生成可靠的UUID对于数据追踪、会话管理、缓存键生成等场景至关重要。接下来将深入探讨UUID的实现原理、前端生成方案及最佳实践。 一、UUID标准与版本 1. UUID结构…...
玩客云WS1608控制LED灯的颜色
玩客云WS1608控制LED灯的颜色 玩客云设备有个红、绿、蓝三色led灯,在刷入armbian系统以后,这个灯的颜色就会显示异常,往往是一直显示红色。 如果要自动动手调整led灯的颜色,控制命令如下(需要root用户执行࿰…...

实验三 企业网络搭建及应用
实验三 企业网络搭建及应用 一、实验目的 1.掌握企业网络组建方法。 2.掌握企业网中常用网络技术配置方法。 二、实验描述 某企业设有销售部、市场部、技术部和财务部四个部门。公司内部网络使用二层交换机作为用户的接入设备。为了使网络更加稳定可靠,公司决定…...

顶会新热门:机器学习可解释性
🧀机器学习模型的可解释性一直是研究的热点和挑战之一,同样也是近两年各大顶会的投稿热门。 🧀这是因为模型的决策过程不仅需要高准确性,还需要能被我们理解,不然我们很难将它迁移到其它的问题中,也很难进…...
ReactJS 中的 JSX工作原理
文章目录 前言✅ 1. JSX 是什么?🔧 2. 编译后的样子(核心机制)🧱 3. React.createElement 做了什么?🧠 4. JSX 与组件的关系🔄 5. JSX 到真实 DOM 的过程📘 6. JSX 与 Fr…...

《STL--stack 和 queue 的使用及其底层实现》
引言: 上次我们学习了容器list的使用及其底层实现,相对来说是比较复杂的,今天我们要学习的适配器stack和queue与list相比就简单很多了,下面我们就开始今天的学习: 一:stack(后进先出ÿ…...
ArcGIS Pro 3.4 二次开发 - 地理处理
环境:ArcGIS Pro SDK 3.4 + .NET 8 文章目录 地理处理1 通用1.1 如何执行模型工具1.2 设置地理处理范围环境1.3 在 Geoprocessing 窗格中打开脚本工具对话框1.4 打开特定工具的地理处理工具窗格1.5 获取地理处理项目项1.6 阻止通过GP创建的特征类自动添加到地图中1.7 GPExecut…...

基于springboot的医护人员排班系统设计与实现(源码+文档+部署讲解)
技术范围:SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容:免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论文…...
Asp.Net Core FluentValidation校验框架
文章目录 前言一、使用步骤1.安装 NuGet 包2.创建模型3.创建验证器4.配置 Program.cs5.创建控制器6.测试结果 二、常见问题及注意事项三、性能优化建议总结 前言 FluentValidation 是一个流行的 .NET 库,用于构建强类型的验证规则。它通常用于验证领域模型、DTO等对…...

CRISPR-Cas系统的小型化研究进展-文献精读137
Progress in the miniaturization of CRISPR-Cas systems CRISPR-Cas系统的小型化研究进展 摘要 CRISPR-Cas基因编辑技术由于其简便性和高效性,已被广泛应用于生物学、医学、农学等领域的基础与应用研究。目前广泛使用的Cas核酸酶均具有较大的分子量(通…...

利用python工具you-get下载网页的视频文件
有时候我们可能在一个网站看到一个视频(比如B站),想下载,但是页面没有下载视频的按钮。这时候,我们可以借助python工具you-get来实现下载功能。下面简要说下步骤 (一)因为使用的是python工具&a…...
Wi-Fi 切换 5G 的时机
每天都希望 Wi-Fi 在我离开信号覆盖范围时能尽快切到 5G,但每次它都能坚挺到最后半格信号,我却连看个天气预报都看不了…我不得不手工关闭 Wi-Fi,然后等走远了之后再打开,如此反复,不厌其烦。 早上出门上班,…...
【请关注】各类数据库优化,抓大重点整改,快速优化空间mysql,Oracle,Neo4j等
各类数据库优化,抓大重点整改,快速优化,首先分析各数据库查询全部表的空间大小及记录条数的语句: MySQL -- 查看所有表的空间大小 SELECT TABLE_SCHEMA AS 数据库名, TABLE_NAME AS 表名, ENGINE AS 存储引擎, CONCAT(ROUND(DAT…...
Mybatis Plus JSqlParser解析sql语句及JSqlParser安装步骤
MyBatis Plus 整合 JSqlParser 进行 SQL 解析的实现方案,主要包括环境配置和具体应用。通过 Maven 添加mybatis-plus-core 和 jsqlparser 依赖后,可用 CCJSqlParserUtil 解析 SQL 语句,支持对 SELECT、UPDATE 等语句的语法树分析和重构。技术…...
React从基础入门到高级实战:React 高级主题 - 性能优化:深入探索与实践指南
React 性能优化:深入探索与实践指南 引言 在现代Web开发中,尤其是2025年的技术环境下,React应用的性能优化已成为开发者不可忽视的核心课题。随着用户对应用速度和体验的要求日益提高,React应用的规模和复杂性不断增加ÿ…...
负载均衡群集---Haproxy
目录 一、HAproxy 一、概念 二、核心作用 三、主要功能特性 四、应用场景 五、优势与特点 二、 案例分析 1. 案例概述 2. 案例前置知识点 (1)HTTP 请求 (2)负载均衡常用调度算法 (3)常见的 web …...
2025年5月个人工作生活总结
本文为 2025年5月工作生活总结。 研发编码 一个项目的临时记录 月初和另一项目同事向业主汇报方案,两个项目都不满意,后来领导做了调整,将项目合并,拆分了好几大块。原来我做的一些工作,如数据库、中间件等ÿ…...

【stm32开发板】单片机最小系统原理图设计
一、批量添加网络标签 可以选择浮动工具中的N,单独为引脚添加网络标签。 当芯片引脚非常多的时候,选中芯片,右键选择扇出网络标签/非连接标识 按住ctrl键即可选中多个引脚 点击将引脚名称填入网络名 就完成了引脚标签的批量添加 二、电源引…...

实验设计与分析(第6版,Montgomery)第5章析因设计引导5.7节思考题5.2 R语言解题
本文是实验设计与分析(第6版,Montgomery著,傅珏生译) 第5章析因设计引导5.7节思考题5.2 R语言解题。主要涉及方差分析,正态假设检验,残差分析,交互作用。 dataframe<-data.frame( Surfacec(74,64,60,92…...

2025山东CCPC题解
文章目录 L - StellaD - Distributed SystemI - Square PuzzleE - Greatest Common DivisorG - Assembly Line L - Stella 题目来源:L - Stella 解题思路 签到题,因为给出的字母不是按顺序,可以存起来赋其值,然后在比较。 代码…...
【解决办法】ubuntu重启不起来,输入用户名和密码进不去,又重新返回登录页。
项目场景: ubuntu重启不起来,输入用户名和密码进不去,又重新返回登录页。 问题描述 在华硕天选一代笔记本上面安装了ubuntu22.04.5桌面版,但是重启以后出现,输入了用户名和密码,等待一会还让输入用户名和…...

CentOS Stream 9 中部署 MySQL 8.0 MGR(MySQL Group Replication)一主两从高可用集群
🐇明明跟你说过:个人主页 🏅个人专栏:《MySQL技术精粹》🏅 🔖行路有良友,便是天堂🔖 目录 一、前言 1、MySQL 8.0 中的高可用方案 2、适用场景 二、环境准备 1、系统环境说明…...

pycharm 新UI 固定菜单栏 pycharm2025 中文版
pycharm 新UI 文件 -> 设置 -> 外观与行为 -> 外观 -> UI选项 -> 主菜单:显示在主工具栏上方. 即可固定...
跟单业务和量化交易业务所涉及到的设计模式
🔁 跟单业务中常用的设计模式: 1. 观察者模式(Observer) 场景:一个大V下单,系统需要自动通知所有跟随者进行同步下单。好处:解耦下单者与跟随者,支持灵活扩展、异步通知。面试亮点…...

我的世界Java版1.21.4的Fabric模组开发教程(十一)创建方块
这是适用于Minecraft Java版1.21.4的Fabric模组开发系列教程专栏第十一章——创建方块。想要阅读其他内容,请查看或订阅上面的专栏。 方块(Block) 是构成Minecraft世界的主要组成部分,是组成游戏地图的最基本单元,也是模组开发的核心元素之一…...