maven

<!-- https://mvnrepository.com/artifact/net.coobird/thumbnailator -->
        <dependency>
            <groupId>net.coobird</groupId>
            <artifactId>thumbnailator</artifactId>
            <version>0.4.11</version>
        </dependency>

Thumbnailator 是一个优秀的图片处理的Google开源Java类库。处理效果远比Java API的好。从API提供现有的图像文件和图像对象的类中简化了处理过程,两三行代码就能够从现有图片生成处理后的图片,且允许微调图片的生成方式,同时保持了需要写入的最低限度的代码量。还支持对一个目录的所有图片进行批量处理操作。

支持的处理操作:图片缩放,区域裁剪,水印,旋转,保持比例。

下面将展示部分常用方法 

import net.coobird.thumbnailator.Thumbnails;
import net.coobird.thumbnailator.geometry.Positions;

import javax.imageio.ImageIO;
import java.io.File;
import java.io.IOException;

public class Picture {

    public static void main(String[] args) throws IOException {
        //图片路径
        String imagePath = "D:\\tmp\\images\\test.png";
        //保存生成图片路径
        String genePath = "D:\\tmp\\images\\geneTest.jpg";
        //将指定文件压缩后 放置到指定路径下
        //下面是一些常用的方法展示
        Thumbnails.of(imagePath)
                .scale(0.5f)  //按比例放大缩小  >0   cale(scaleWidth,scaleHeight)
//                .size(820, 547)  //设置生成图片的长宽 与scale 不能同时使用
                .rotate(90)    rotate(角度),正数:顺时针负数:逆时针
                watermark(位置,水印图,透明度)
                .watermark(Positions.CENTER, ImageIO.read(new File("D:\\tmp\\images\\watermark.png")),0.5f)    
                .outputFormat("jpg")   //保存为文件的格式设置
                .allowOverwrite(true) //是否允许覆盖已存在的文件
                .outputQuality(0.5f)    //输出的图片质量  0~1 之间,否则报错
//                .toOutputStream(new FileOutputStream("url"))   //输出到指定的输出流中
                   // //asBufferedImage()返回BufferedImage  ImageIO.write(thumbnail,"jpg",newFile("c:/a380_1280x1024_BufferedImage.jpg"));
//                .asBufferedImages()     
                .toFile(genePath);
    }
}

展示案例

添加了水印、旋转、缩小、压缩

java图片压缩 exif信息丢失 java图片压缩工具_放大缩小

可以明显看到压缩前文件时1.71MB,压缩后的文件大小为13.5KB(压缩的有点凶,但效果还不错)

java图片压缩 exif信息丢失 java图片压缩工具_Image_02

场景使用

场景一:图片尺寸不变,修改图片文件类型

//图片路径
        String imagePath = "D:\\tmp\\images\\test.png";
        //保存生成图片路径
        String genePath = "D:\\tmp\\images\\geneTest.jpg";
        //改变文件格式
        Thumbnails.of(imagePath)
                .scale(1f) //按比例放大缩小 和size() 必须使用一个 不然会报错
                .outputFormat("jpg")   //保存为文件的格式设置
                .toFile(genePath);

场景二:图片尺寸不变,压缩图片文件大小(避免网页上的图片过大,占用过多的带宽)

//图片路径
        String imagePath = "D:\\tmp\\images\\test.png";
        //保存生成图片路径
        String genePath = "D:\\tmp\\images\\geneTest.jpg";
        //改变文件格式
        Thumbnails.of(imagePath)
                .scale(1f) //按比例放大缩小 和size() 必须使用一个 不然会报错
                .outputQuality(0.5f)    //输出的图片质量  0~1 之间,否则报错
                .outputFormat("jpg")   //保存为文件的格式设置 可不设置
                .toFile(genePath);

注 : 图片的质量 参数设置 必须在0~1 否则会报错

场景三:压缩至指定图片尺寸(例如:横400高300),不保持图片比例

//图片路径
        String imagePath = "D:\\tmp\\images\\test.png";
        //保存生成图片路径
        String genePath = "D:\\tmp\\images\\geneTest.jpg";
        //改变文件格式
        Thumbnails.of(imagePath)
                .forceSize(400, 100)  //和size() 区别在于keepAspectRatio = false,不会按比例收缩,允许图片变形
                .toFile(genePath);

场景四:压缩至指定图片尺寸(例如:横400高300),保持图片不变形,多余部分裁剪掉

//图片路径
        String imagePath = "D:\\tmp\\images\\test.png";
        //保存生成图片路径
        String genePath = "D:\\tmp\\images\\geneTest.jpg";
        
        BufferedImage image = ImageIO.read(new File(imagePath));
        Builder<BufferedImage> builder = null;

        int imageWidth = image.getWidth();
        int imageHeitht = image.getHeight();
        if ((float)300 / 400 != (float)imageWidth / imageHeitht) {
            if (imageWidth > imageHeitht) {
                image = Thumbnails.of(imagePath).height(300).asBufferedImage();
            } else {
                image = Thumbnails.of(imagePath).width(400).asBufferedImage();
            }
            builder = Thumbnails.of(image).sourceRegion(Positions.CENTER, 400, 300).size(400, 300);
        } else {
            builder = Thumbnails.of(image).size(400, 300);
        }
        builder.outputFormat("jpg").toFile(genePath);

这种情况复杂些,既不能用size()方法(因为横高比不一定是4/3,这样压缩后的图片横为400高为300),也不能用forceSize()方法。首先判断横高比,确定是按照横400压缩还是高300压缩,压缩后按中心400*300的区域进行裁剪,这样得到的图片便是400*300的裁剪后缩略图。

使用size()或forceSize()方法时,如果图片比指定的尺寸要小(比如size(400, 300),而图片为40*30),则会拉伸到指定尺寸。

总体来说,该框架是非常优秀的