原文鏈接
https://my.oschina.net/u/242764/blog/2980538
https://blog.csdn.net/Abubu123/article/details/77099950
package com.bonc.ioc.lmp.common.util; import java.awt.Graphics2D; import java.awt.Rectangle; import java.awt.RenderingHints; import java.awt.geom.AffineTransform; import java.awt.image.BufferedImage; import java.awt.image.ColorModel; import java.awt.image.WritableRaster; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.IOException; import java.io.InputStream; import javax.imageio.ImageIO; import org.junit.Test; /** * 圖片工具類,完成圖片的截取 * 所有方法返回值均未boolean型 */ public class ImageHelper { /** * 實現圖像的等比縮放 * @param source * @param targetW * @param targetH * @return */ private static BufferedImage resize(BufferedImage source, int targetW, int targetH) { // targetW,targetH分別表示目標長和寬 int type = source.getType(); BufferedImage target = null; double sx = (double) targetW / source.getWidth(); double sy = (double) targetH / source.getHeight(); // 這里想實現在targetW,targetH范圍內實現等比縮放。如果不需要等比縮放 // 則將下面的if else語句注釋即可 if (sx < sy) { sx = sy; targetW = (int) (sx * source.getWidth()); } else { sy = sx; targetH = (int) (sy * source.getHeight()); } if (type == BufferedImage.TYPE_CUSTOM) { // handmade ColorModel cm = source.getColorModel(); WritableRaster raster = cm.createCompatibleWritableRaster(targetW, targetH); boolean alphaPremultiplied = cm.isAlphaPremultiplied(); target = new BufferedImage(cm, raster, alphaPremultiplied, null); } else target = new BufferedImage(targetW, targetH, type); Graphics2D g = target.createGraphics(); // smoother than exlax: g.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BICUBIC); g.drawRenderedImage(source, AffineTransform.getScaleInstance(sx, sy)); g.dispose(); return target; } /** * 實現圖像的等比縮放和縮放后的截取, 處理成功返回true, 否則返回false * @param inFilePath 要截取文件的路徑 * @param outFilePath 截取后輸出的路徑 * @param width 要截取寬度 * @param hight 要截取的高度 * @throws Exception */ public static boolean compress(String inFilePath, String outFilePath, int width, int hight) { boolean ret = false; File file = new File(inFilePath); File saveFile = new File(outFilePath); InputStream in = null; try { in = new FileInputStream(file); ret = compress(in, saveFile, width, hight); } catch (FileNotFoundException e) { e.printStackTrace(); ret = false; } finally{ if(null != in){ try { in.close(); } catch (IOException e) { e.printStackTrace(); } } } return ret; } /** * 實現圖像的等比縮放和縮放后的截取, 處理成功返回true, 否則返回false * @param in 要截取文件流 * @param outFilePath 截取后輸出的路徑 * @param width 要截取寬度 * @param hight 要截取的高度 * @throws Exception */ public static boolean compress(InputStream in, File saveFile, int width, int hight) { // boolean ret = false; BufferedImage srcImage = null; try { srcImage = ImageIO.read(in); } catch (IOException e) { e.printStackTrace(); return false; } if (width > 0 || hight > 0) { // 原圖的大小 int sw = srcImage.getWidth(); int sh = srcImage.getHeight(); // 如果原圖像的大小小於要縮放的圖像大小,直接將要縮放的圖像復制過去 if (sw > width && sh > hight) { srcImage = resize(srcImage, width, hight); } else { String fileName = saveFile.getName(); String formatName = fileName.substring(fileName .lastIndexOf('.') + 1); try { ImageIO.write(srcImage, formatName, saveFile); } catch (IOException e) { e.printStackTrace(); return false; } return true; } } // 縮放后的圖像的寬和高 int w = srcImage.getWidth(); int h = srcImage.getHeight(); // 如果縮放后的圖像和要求的圖像寬度一樣,就對縮放的圖像的高度進行截取 if (w == width) { // 計算X軸坐標 int x = 0; int y = h / 2 - hight / 2; try { saveSubImage(srcImage, new Rectangle(x, y, width, hight), saveFile); } catch (IOException e) { e.printStackTrace(); return false; } } // 否則如果是縮放后的圖像的高度和要求的圖像高度一樣,就對縮放后的圖像的寬度進行截取 else if (h == hight) { // 計算X軸坐標 int x = w / 2 - width / 2; int y = 0; try { saveSubImage(srcImage, new Rectangle(x, y, width, hight), saveFile); } catch (IOException e) { e.printStackTrace(); return false; } } return true; } /** * 實現圖像的等比縮放和縮放后的截取, 處理成功返回true, 否則返回false * @param in 圖片輸入流 * @param saveFile 壓縮后的圖片輸出流 * @param proportion 壓縮比 * @throws Exception */ public static boolean compress(InputStream in, File saveFile, int proportion) { if(null == in ||null == saveFile ||proportion < 1){// 檢查參數有效性 //LoggerUtil.error(ImageHelper.class, "--invalid parameter, do nothing!"); return false; } BufferedImage srcImage = null; try { srcImage = ImageIO.read(in); } catch (IOException e) { e.printStackTrace(); return false; } // 原圖的大小 int width = srcImage.getWidth() / proportion; int hight = srcImage.getHeight() / proportion; srcImage = resize(srcImage, width, hight); // 縮放后的圖像的寬和高 int w = srcImage.getWidth(); int h = srcImage.getHeight(); // 如果縮放后的圖像和要求的圖像寬度一樣,就對縮放的圖像的高度進行截取 if (w == width) { // 計算X軸坐標 int x = 0; int y = h / 2 - hight / 2; try { saveSubImage(srcImage, new Rectangle(x, y, width, hight), saveFile); } catch (IOException e) { e.printStackTrace(); return false; } } // 否則如果是縮放后的圖像的高度和要求的圖像高度一樣,就對縮放后的圖像的寬度進行截取 else if (h == hight) { // 計算X軸坐標 int x = w / 2 - width / 2; int y = 0; try { saveSubImage(srcImage, new Rectangle(x, y, width, hight), saveFile); } catch (IOException e) { e.printStackTrace(); return false; } } return true; } /** * 實現縮放后的截圖 * @param image 縮放后的圖像 * @param subImageBounds 要截取的子圖的范圍 * @param subImageFile 要保存的文件 * @throws IOException */ private static void saveSubImage(BufferedImage image, Rectangle subImageBounds, File subImageFile) throws IOException { if (subImageBounds.x < 0 || subImageBounds.y < 0 || subImageBounds.width - subImageBounds.x > image.getWidth() || subImageBounds.height - subImageBounds.y > image.getHeight()) { //LoggerUtil.error(ImageHelper.class, "Bad subimage bounds"); return; } BufferedImage subImage = image.getSubimage(subImageBounds.x,subImageBounds.y, subImageBounds.width, subImageBounds.height); String fileName = subImageFile.getName(); String formatName = fileName.substring(fileName.lastIndexOf('.') + 1); ImageIO.write(subImage, formatName, subImageFile); } @Test public static void main(String[] args) throws Exception { /** * saveSubImage 截圖類的使用 * srcImage 為BufferedImage對象 * Rectangle 為需要截圖的長方形坐標 * saveFile 需要保存的路徑及名稱 * **/ //需要截圖的長方形坐標 /*Rectangle rect =new Rectangle(); rect.x=40; rect.y=40; rect.height=160; rect.width=160; InputStream in = null; //需要保存的路徑及名稱 File saveFile = new File("d:\\ioc\\files\\aaa2.jpg"); //需要進行處理的圖片的路徑 in = new FileInputStream(new File("d:\\ioc\\files\\aaa.jpg")); BufferedImage srcImage = null; //將輸入的數據轉為BufferedImage對象 srcImage = ImageIO.read(in); ImageHelper img=new ImageHelper(); img.saveSubImage(srcImage, rect, saveFile);*/ /** * compress 圖片縮放類的使用(縮略圖) * srcImage 為InputStream對象 * Rectangle 為需要截圖的長方形坐標 * proportion 為壓縮比例 * **/ InputStream in = null; //縮放后需要保存的路徑 File saveFile = new File("d:\\ioc\\files\\aaa2.jpg"); try { //原圖片的路徑 in = new FileInputStream(new File("d:\\ioc\\files\\aaa.jpg")); if(compress(in, saveFile, 10)){ System.out.println("圖片壓縮十倍!"); } } catch (Exception e) { e.printStackTrace(); } finally { in.close(); } } }
保留圖片大小,壓縮圖片精度
import javax.imageio.IIOImage; import javax.imageio.ImageIO; import javax.imageio.ImageWriteParam; import javax.imageio.ImageWriter; import java.awt.image.BufferedImage; import java.awt.image.ColorModel; import java.io.*; import java.util.Iterator; /** * @author: Supermap·F * @descripition: 圖片壓縮,保留圖片尺寸大小,壓縮精度 * @date: created in 18:46 2020/8/20 * @modify: Copyright (c) Supermap All Rights Reserved. */ public class ImgUtil { public static byte[] compressPictureByQality(byte[] imgByte, float qality){ byte[] imgBytes = null; BufferedImage resultImage = null; try { ByteArrayInputStream byteInput = new ByteArrayInputStream(imgByte); BufferedImage image = ImageIO.read(byteInput); //如果圖片為空,返回null if (image == null){ return null; } //得到指定的format圖片的writer(迭代器) Iterator<ImageWriter> iter = ImageIO.getImageWritersByFormatName("jpeg"); //得到writer ImageWriter writer = (ImageWriter) iter.next(); //得到指定writer的輸出參數設置 ImageWriteParam iwp = writer.getDefaultWriteParam(); //設置是否可以壓縮 iwp.setCompressionMode(ImageWriteParam.MODE_EXPLICIT); //設置壓縮質量參數 iwp.setCompressionQuality(qality); iwp.setProgressiveMode(ImageWriteParam.MODE_DISABLED); ColorModel colorModel = ColorModel.getRGBdefault(); //指定壓縮式使用的色彩模式 iwp.setDestinationType(new javax.imageio.ImageTypeSpecifier(colorModel,colorModel.createCompatibleSampleModel(16,16))); //開始打包圖片,寫入byte[] //取的內存輸出流 ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); IIOImage iImage = new IIOImage(image, null, null); //ImageWriter 中用來接收writer信息的output要求必須是ImageOutput writer.setOutput(ImageIO.createImageOutputStream(byteArrayOutputStream)); writer.write(null, iImage, iwp); imgBytes = byteArrayOutputStream.toByteArray(); ByteArrayInputStream byteInputs = new ByteArrayInputStream(imgBytes); resultImage = ImageIO.read(byteInputs); }catch (IOException e){ System.out.println("錯誤"); e.printStackTrace(); } return imgBytes; } /** * 圖片壓縮 * @param image 圖片 * @param qality * @return * @throws IOException */ public static BufferedImage compressPictureByQalityV2(BufferedImage image, float qality) { byte[] imgBytes = null; BufferedImage resultImage = null; try { // ByteArrayInputStream byteInput = new ByteArrayInputStream(imgByte); // BufferedImage image = ImageIO.read(byteInput); //如果圖片為空,返回null if (image == null){ return null; } //得到指定的format圖片的writer(迭代器) Iterator<ImageWriter> iter = ImageIO.getImageWritersByFormatName("jpeg"); //得到writer ImageWriter writer = (ImageWriter) iter.next(); //得到指定writer的輸出參數設置 ImageWriteParam iwp = writer.getDefaultWriteParam(); //設置是否可以壓縮 iwp.setCompressionMode(ImageWriteParam.MODE_EXPLICIT); //設置壓縮質量參數 iwp.setCompressionQuality(qality); iwp.setProgressiveMode(ImageWriteParam.MODE_DISABLED); ColorModel colorModel = ColorModel.getRGBdefault(); //指定壓縮式使用的色彩模式 iwp.setDestinationType(new javax.imageio.ImageTypeSpecifier(colorModel,colorModel.createCompatibleSampleModel(16,16))); //開始打包圖片,寫入byte[] //取的內存輸出流 ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); IIOImage iImage = new IIOImage(image, null, null); //ImageWriter 中用來接收writer信息的output要求必須是ImageOutput writer.setOutput(ImageIO.createImageOutputStream(byteArrayOutputStream)); writer.write(null, iImage, iwp); imgBytes = byteArrayOutputStream.toByteArray(); ByteArrayInputStream byteInputs = new ByteArrayInputStream(imgBytes); resultImage = ImageIO.read(byteInputs); }catch (IOException e){ System.out.println("錯誤"); e.printStackTrace(); } return resultImage; } }