JAVA (Graphics2D)解决合成图片失真问题

2023-10-06 04:30

本文主要是介绍JAVA (Graphics2D)解决合成图片失真问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

附对比图:

合成后失真严重

代码优化后:

原代码:

参数解释:

1. img为你需要把他合成到A图片上的B图片,以下简称为章

2. srcImagePath为A图片的绝对路径

3. targetPath为合成后的图片输出路径

4. x,y为章需要合成到A图片上的位置坐标,w,h为将章缩小到多少长宽比例

注意这行代码:g.drawImage(img, x, y, w, h, null);

public static void markByIcon(Image img, String srcImagePath, String targetPath, int x, int y,int w, int h) throws IOException {OutputStream os = null;try {logger.info("图片水印开始添加。。。");logger.info("图片水印输入路径" + srcImagePath);logger.info("图片水印输出路径" + targetPath);Image srcImage = ImageIO.read(new File(srcImagePath));BufferedImage buffImg = new BufferedImage(srcImage.getWidth(null), srcImage.getHeight(null),BufferedImage.TYPE_INT_BGR);Graphics2D g = buffImg.createGraphics();g.setRenderingHint(RenderingHints.KEY_INTERPOLATION,RenderingHints.VALUE_INTERPOLATION_BILINEAR);g.drawImage(srcImage.getScaledInstance(srcImage.getWidth(null), srcImage.getHeight(null), Image.SCALE_SMOOTH),0, 0, null);g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, 1));g.drawImage(img, x, y, w, h, null);g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER));g.dispose();os = new FileOutputStream(targetPath);//生成图片ImageIO.write(buffImg, "JPG", os);logger.info("图片水印添加成功。。。");} catch (Exception e) {logger.error("图片水印添加失败。。。", e);throw e;} finally {try {if (null != os) {os.close();}} catch (Exception e) {e.printStackTrace();}}}

优化后代码:

原代码是将章在合成的时候按比例缩小,在缩小过程中失真了,优化后为先将章缩小到想要的比例,然后用缩小后的章进行合成

压缩代码:

/*** * 将图片按照指定的图片尺寸压缩 * * @param srcImgPath :源图片路径 * @param outImgPath * :输出的压缩图片的路径 * @param new_w* * :压缩后的图片宽 * @param new_h * :压缩后的图片高*/public static BufferedImage compressImage(String srcImgPath, int new_w, int new_h) {BufferedImage src = inputImage(srcImgPath);BufferedImage bufferedImage = disposeImage(src, new_w, new_h);return bufferedImage;}/*** 处理图片 * * @param src * @param outImgPath * @param new_w * @param new_h*/private synchronized static BufferedImage disposeImage(BufferedImage src, int new_w, int new_h) {// 得到图片int old_w = src.getWidth();// 得到源图宽int old_h = src.getHeight();// 得到源图长BufferedImage newImg = null;// 判断输入图片的类型switch (src.getType()) {case 13:// png,gifnewImg = new BufferedImage(new_w, new_h,// BufferedImage.TYPE_4BYTE_ABGR);break;default:newImg = new BufferedImage(new_w, new_h, BufferedImage.TYPE_INT_RGB);break;}Graphics2D g = newImg.createGraphics();// 从原图上取颜色绘制新图g.drawImage(src, 0, 0, old_w, old_h, null);g.dispose();// 根据图片尺寸压缩比得到新图的尺寸newImg.getGraphics().drawImage(src.getScaledInstance(new_w, new_h, Image.SCALE_SMOOTH), 0, 0,null);// 调用方法输出图片文件return newImg;}/*** * 将图片文件输出到指定的路径,并可设定压缩质量 * * @param outImgPath * @param newImg * @param per*/private static void OutImage(String outImgPath, BufferedImage newImg) {// 判断输出的文件夹路径是否存在,不存在则创建File file = new File(outImgPath);if (!file.getParentFile().exists()) {file.getParentFile().mkdirs();}// 输出到文件流try {ImageIO.write(newImg, outImgPath.substring(outImgPath.lastIndexOf(".") + 1), new File(outImgPath));} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

优化后代码:

先通过compressImage方法将章图片进行比例缩小,然后用缩小后的bufferedImage去进行合成

BufferedImage bufferedImage = ImageUtil.compressImage(signPath, photoConfig.getWidth(), photoConfig.getHeight());
Image signImg = ImageUtil.sign(bufferedImage);
if (signImg == null) {throw new MyException("图片生成失败,签名异常");
}
ImageUtil.markBySignIcon(signImg, imagePath, targetPath, x, y, photoConfig.getWidth(), photoConfig.getHeight());

此时我将原markByIcon方法进行了修改,修改后代码如下,修改后的方法名为markBySignIcon如上最后一行

public static void markBySignIcon(Image img, String srcImagePath, String targetPath, int x, int y,int w, int h) throws IOException {OutputStream os = null;try {logger.info("图片水印开始添加。。。");logger.info("图片水印输入路径" + srcImagePath);logger.info("图片水印输出路径" + targetPath);Image srcImage = ImageIO.read(new File(srcImagePath));BufferedImage buffImg = new BufferedImage(srcImage.getWidth(null), srcImage.getHeight(null),BufferedImage.TYPE_INT_BGR);Graphics2D g = buffImg.createGraphics();g.setRenderingHint(RenderingHints.KEY_INTERPOLATION,RenderingHints.VALUE_INTERPOLATION_BILINEAR);g.drawImage(srcImage.getScaledInstance(srcImage.getWidth(null), srcImage.getHeight(null), Image.SCALE_SMOOTH),0, 0, null);g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, 1));g.drawImage(img, x, y, img.getWidth(null), img.getHeight(null), null);g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER));g.dispose();os = new FileOutputStream(targetPath);//生成图片ImageIO.write(buffImg, "JPG", os);logger.info("图片水印添加成功。。。");} catch (Exception e) {logger.error("图片水印添加失败。。。", e);throw e;} finally {try {if (null != os) {os.close();}} catch (Exception e) {e.printStackTrace();}}}

还记得之前标红,要注意的代码么,现在改成如下

g.drawImage(img, x, y, img.getWidth(null), img.getHeight(null), null);

到此,用来合成的小图章在合成后,效果会较原来好很多

这篇关于JAVA (Graphics2D)解决合成图片失真问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/152833

相关文章

Java中Arrays类和Collections类常用方法示例详解

《Java中Arrays类和Collections类常用方法示例详解》本文总结了Java中Arrays和Collections类的常用方法,涵盖数组填充、排序、搜索、复制、列表转换等操作,帮助开发者高... 目录Arrays.fill()相关用法Arrays.toString()Arrays.sort()A

Spring Boot Maven 插件如何构建可执行 JAR 的核心配置

《SpringBootMaven插件如何构建可执行JAR的核心配置》SpringBoot核心Maven插件,用于生成可执行JAR/WAR,内置服务器简化部署,支持热部署、多环境配置及依赖管理... 目录前言一、插件的核心功能与目标1.1 插件的定位1.2 插件的 Goals(目标)1.3 插件定位1.4 核

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

Java堆转储文件之1.6G大文件处理完整指南

《Java堆转储文件之1.6G大文件处理完整指南》堆转储文件是优化、分析内存消耗的重要工具,:本文主要介绍Java堆转储文件之1.6G大文件处理的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言文件为什么这么大?如何处理这个文件?分析文件内容(推荐)删除文件(如果不需要)查看错误来源如何避

SpringBoot整合Dubbo+ZK注册失败的坑及解决

《SpringBoot整合Dubbo+ZK注册失败的坑及解决》使用Dubbo框架时,需在公共pom添加依赖,启动类加@EnableDubbo,实现类用@DubboService替代@Service,配... 目录1.先看下公共的pom(maven创建的pom工程)2.启动类上加@EnableDubbo3.实

SpringBoot整合(ES)ElasticSearch7.8实践

《SpringBoot整合(ES)ElasticSearch7.8实践》本文详细介绍了SpringBoot整合ElasticSearch7.8的教程,涵盖依赖添加、客户端初始化、索引创建与获取、批量插... 目录SpringBoot整合ElasticSearch7.8添加依赖初始化创建SpringBoot项

JAVA覆盖和重写的区别及说明

《JAVA覆盖和重写的区别及说明》非静态方法的覆盖即重写,具有多态性;静态方法无法被覆盖,但可被重写(仅通过类名调用),二者区别在于绑定时机与引用类型关联性... 目录Java覆盖和重写的区别经常听到两种话认真读完上面两份代码JAVA覆盖和重写的区别经常听到两种话1.覆盖=重写。2.静态方法可andro

Kotlin Map映射转换问题小结

《KotlinMap映射转换问题小结》文章介绍了Kotlin集合转换的多种方法,包括map(一对一转换)、mapIndexed(带索引)、mapNotNull(过滤null)、mapKeys/map... 目录Kotlin 集合转换:map、mapIndexed、mapNotNull、mapKeys、map

nginx中端口无权限的问题解决

《nginx中端口无权限的问题解决》当Nginx日志报错bind()to80failed(13:Permissiondenied)时,这通常是由于权限不足导致Nginx无法绑定到80端口,下面就来... 目录一、问题原因分析二、解决方案1. 以 root 权限运行 Nginx(不推荐)2. 为 Nginx