1、背景
在开发过程中,我们经常需要把大图片先进行压缩,再上传至服务器。
2、介绍
Thumbnailator是一个Google开源的优秀图片处理的第三方Java类库,处理效果远比Java API的好。
Thumbnailator可以使用很少的代码实现图片的压缩功能,同时也提供了图片缩放、旋转与加水印等功能,本文目前只介绍压缩。
3、实现
3.1、maven依赖
<!--图片压缩-->
<dependency>
<groupId>net.coobird</groupId>
<artifactId>thumbnailator</artifactId>
<version>0.4.8</version>
</dependency>
3.2、压缩方法一(压缩后转成输出流)不推荐
不推荐的原因是在处理一些图片时,不能将图片压缩,反而会使图片增大
class ImageUtils {
private static final Logger log = LoggerFactory.getLogger(ImageUtils.class);
private static final Integer ZERO = 0;
private static final Integer ONE_ZERO_TWO_FOUR = 1024;
private static final Integer NINE_ZERO_ZERO = 900;
private static final Integer THREE_TWO_SEVEN_FIVE = 3275;
private static final Integer TWO_ZERO_FOUR_SEVEN = 2047;
private static final Double ZERO_EIGHT_FIVE = 0.85;
private static final Double ZERO_SIX = 0.6;
private static final Double ZERO_FOUR_FOUR = 0.44;
private static final Double ZERO_FOUR = 0.4;
private static final String JPG = ".jpg";
/**
* 根据指定大小压缩图片
*
* @param imageBytes 源图片字节数组
* @param desFileSize 指定图片大小,单位kb
* @return 压缩质量后的图片字节数组
* scale 0~1 缩放倍数,1表示不缩放,但是这个方法指定不缩放,会出现死循环
*/
public static byte[] compressPicForScale(byte[] imageBytes, long desFileSize) {
if (imageBytes == null || imageBytes.length <= ZERO || imageBytes.length < desFileSize * ONE_ZERO_TWO_FOUR) {
return imageBytes;
}
long srcSize = imageBytes.length;
double accuracy = getAccuracy(srcSize / ONE_ZERO_TWO_FOUR);
try {
byte[] bytes = imageBytes;
long start = System.currentTimeMillis();
while (imageBytes.length > desFileSize * ONE_ZERO_TWO_FOUR) {
ByteArrayInputStream inputStream = new ByteArrayInputStream(imageBytes);
ByteArrayOutputStream outputStream = new ByteArrayOutputStream(imageBytes.length);
Thumbnails.of(inputStream)
.scale(accuracy)
.outputQuality(accuracy)
.toOutputStream(outputStream);
imageBytes = outputStream.toByteArray();
long end = System.currentTimeMillis();
if (end - start > 5 * 1000L){
// 如果5秒钟不能完成,则返回原图片,避免死循环。
return bytes;
}
}
log.info("图片原大小={}kb | 压缩后大小={}kb",
srcSize / ONE_ZERO_TWO_FOUR, imageBytes.length / ONE_ZERO_TWO_FOUR);
} catch (Exception e) {
log.error("【图片压缩】msg=图片压缩失败!", e);
}
return imageBytes;
}
/**
* 自动调节精度(经验数值)
*
* @param size 源图片大小
* @return 图片压缩质量比
*/
private static double getAccuracy(long size) {
double accuracy;
if (size < NINE_ZERO_ZERO) {
accuracy = ZERO_EIGHT_FIVE;
} else if (size < TWO_ZERO_FOUR_SEVEN) {
accuracy = ZERO_SIX;
} else if (size < THREE_TWO_SEVEN_FIVE) {
accuracy = ZERO_FOUR_FOUR;
} else {
accuracy = ZERO_FOUR;
}
return accuracy;
}
}
3.3、方法二(压缩后转成文件)
因为要上传服务器,创建本地文件就不太合适了,所以创建一个临时文件
/**
* 根据指定大小压缩图片
*
* @param file 源图片
* @param accuracy 指定图片缩放倍数
* @return 压缩质量后的图片字节数组
* scale 0~1 缩放倍数,1表示不缩放
*/
public static byte[] scalePic(MultipartFile file, double accuracy) throws IOException {
// 压缩图片并保存到临时文件中
File tempFile = File.createTempFile("thumbnail", ".jpg");
Thumbnails.of(file.getInputStream())
.scale(1)
.outputQuality(accuracy)
.toFile(tempFile);
// 读取临时文件的字节流设置到输出流中
InputStream tempInputStream = new FileInputStream(tempFile);
byte[] buffer = new byte[tempInputStream.available()];
tempInputStream.read(buffer);
tempInputStream.close();
// 删除临时文件
tempFile.delete();
// 可以下载到本地看一看,是否真的缩小
// BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("F:\\Code\\upload\\1.jpg"));
// bos.write(buffer);
// bos.close();
return buffer;
}
// 上传图片到第三方服务器,可能需要转成输入流
// bytes 字节数组
InputStream inputStream = new ByteArrayInputStream(bytes);