package com.thinkgem.jeesite.test; import java.awt.Image; import java.awt.image.BufferedImage; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.InputStream; import javax.imageio.ImageIO; import com.sun.image.codec.jpeg.JPEGCodec; import com.sun.image.codec.jpeg.JPEGImageEncoder; @SuppressWarnings("restriction") public class ReduceImgTest { /** * 指定图片宽度和高度或压缩比例对图片进行压缩 * * @param imgsrc 源图片地址 * @param imgdist 目标图片地址 * @param widthdist 压缩后图片的宽度 * @param heightdist 压缩后图片的高度 * @param rate 压缩的比例 */ public static void reduceImg(String imgsrc, String imgdist, int widthdist, int heightdist, Float rate) { try { File srcfile = new File(imgsrc); // 检查图片文件是否存在 if (!srcfile.exists()) { System.out.println("文件不存在"); } // 如果比例不为空则说明是按比例压缩 if (rate != null && rate > 0) { //获得源图片的宽高存入数组中 int[] results = getImgWidthHeight(srcfile); if (results == null || results[0] == 0 || results[1] == 0) { return; } else { //按比例缩放或扩大图片大小,将浮点型转为整型 widthdist = (int) (results[0] * rate); heightdist = (int) (results[1] * rate); } } // 开始读取文件并进行压缩 Image src = ImageIO.read(srcfile); // 构造一个类型为预定义图像类型之一的 BufferedImage BufferedImage tag = new BufferedImage((int) widthdist, (int) heightdist, BufferedImage.TYPE_INT_RGB); //绘制图像 getScaledInstance表示创建此图像的缩放版本,返回一个新的缩放版本Image,按指定的width,height呈现图像 //Image.SCALE_SMOOTH,选择图像平滑度比缩放速度具有更高优先级的图像缩放算法。 tag.getGraphics().drawImage(src.getScaledInstance(widthdist, heightdist, Image.SCALE_SMOOTH), 0, 0, null); //创建文件输出流 FileOutputStream out = new FileOutputStream(imgdist); //将图片按JPEG压缩,保存到out中 JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(out); encoder.encode(tag); //关闭文件输出流 out.close(); } catch (Exception ef) { ef.printStackTrace(); } } /** * 获取图片宽度和高度 * * @param 图片路径 * @return 返回图片的宽度 */ public static int[] getImgWidthHeight(File file) { InputStream is = null; BufferedImage src = null; int result[] = { 0, 0 }; try { // 获得文件输入流 is = new FileInputStream(file); // 从流里将图片写入缓冲图片区 src = ImageIO.read(is); result[0] =src.getWidth(null); // 得到源图片宽 result[1] =src.getHeight(null);// 得到源图片高 is.close(); //关闭输入流 } catch (Exception ef) { ef.printStackTrace(); } return result; } public static void main(String[] args) { File srcfile = new File("C://Users//xsl//Desktop//153.jpg"); File distfile = new File("C://Users//xsl//Desktop//153_2.jpg"); System.out.println("压缩前图片大小:" + srcfile.length()); reduceImg("C://Users//xsl//Desktop//153.jpg", "C://Users//xsl//Desktop//153_2.jpg", 0, 0, 0.5f); System.out.println("压缩后图片大小:" + distfile.length()); /* * 解决这个错误提示 * ERROR: JDWP Unable to get JNI 1.2 environment, jvm->GetEnv() return code = -2 * JDWP exit error AGENT_ERROR_NO_JNI_ENV(183): [../../../src/share/back/util.c:838] */ System.exit(0); } }
Java使用google开源工具实现图片压缩
Thumbnailator,一个google使用的开源的工具类。
这个工具类满足了上面所说的所有的要求。
同时对于图片的处理还有了别的方法,如旋转,裁切,加水印等等。
在github上面的地址是:https://github.com/coobird/thumbnailator
maven的地址
<dependency> <groupId>net.coobird</groupId> <artifactId>thumbnailator</artifactId> <version>0.4.8</version> </dependency>
使用起来特别的简单:一行代码就搞定了
Thumbnails.of("原图文件的路径")
.scale(1f)
.outputQuality(0.5f)
.toFile("压缩后文件的路径");
其中的scale是可以指定图片的大小,值在0到1之间,1f就是原图大小,0.5就是原图的一半大小,这里的大小是指图片的长宽。
而outputQuality是图片的质量,值也是在0到1,越接近于1质量越好,越接近于0质量越差。
对于压缩图片来说上面就已经足够了。
PS:经过使用后的反馈,这个工具无法正确压缩出png格式的图片
因为png本身就是一种无损的图片格式,而jpg是一种压缩的图片格式;
当前方法目的是为了在尽可能不丢失图片质量的情况下进行的压缩;
建议将图片压缩后的格式设置成jpg来解决;.outputFormat("jpg")
工具源码本身最后还是调用jdk中的ImageIO.createImageOutputStream(fos);来实现的;