原文鏈接
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;
}
}
