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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu