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设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

JavaSE正则表达式用法总结大全

《JavaSE正则表达式用法总结大全》正则表达式就是由一些特定的字符组成,代表的是一个规则,:本文主要介绍JavaSE正则表达式用法的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录常用的正则表达式匹配符正则表China编程达式常用的类Pattern类Matcher类PatternSynta

Spring Security中用户名和密码的验证完整流程

《SpringSecurity中用户名和密码的验证完整流程》本文给大家介绍SpringSecurity中用户名和密码的验证完整流程,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 首先创建了一个UsernamePasswordAuthenticationTChina编程oken对象,这是S