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);來實現的;
