java图片压缩工具类

简介: java图片压缩工具类

今天给大家分享已个自己博客在用的图片压缩的一个工具类,压缩效果挺不错的。

import com.sun.image.codec.jpeg.JPEGCodec;
import com.sun.image.codec.jpeg.JPEGEncodeParam;
import com.sun.image.codec.jpeg.JPEGImageEncoder;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.multipart.MultipartFile;

import javax.imageio.ImageIO;
import javax.imageio.ImageReadParam;
import javax.imageio.ImageReader;
import javax.imageio.stream.ImageInputStream;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.*;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.Iterator;

/**
 * @className: ImageUtil
 * @description: 图片压缩工具类
 * @author: xiaofei
 * @create: 2020年01月15日
 */
@Slf4j
public class ImageUtil {

    /**
     * 图片文件读取
     *
     * @param file 文件
     * @return 结果
     */
    private static BufferedImage InputImage(MultipartFile file) throws RuntimeException{
        BufferedImage srcImage = null;
        FileInputStream in = null;
        try {
            // 构造BufferedImage对象
            srcImage = ImageIO.read(file.getInputStream());
        } catch (IOException e) {
            log.error("读取图片文件出错!", e);
            throw new RuntimeException();
        } finally {
            if (in != null) {
                try {
                    in.close();
                } catch (IOException e) {
                    log.error("读取图片文件关闭流出错!", e);
                }
            }
        }
        return srcImage;
    }

    /**
     * 将图片按照指定的图片尺寸、图片质量压缩
     *
     * @param file       :源图片路径
     * @param outImgPath :输出的压缩图片的路径
     * @param new_w      :压缩后的图片宽
     * @param new_h      :压缩后的图片高
     * @param per        :图片质量压缩
     */
    public static void resize(MultipartFile file
            , String outImgPath
            , int new_w
            , int new_h
            , float per) {
        // 得到图片
        BufferedImage src = InputImage(file);
        int old_w = src.getWidth();
        // 得到源图宽
        int old_h = src.getHeight();
        // 得到源图长
        // 根据原图的大小生成空白画布
        BufferedImage tempImg = new BufferedImage(old_w, old_h,
                BufferedImage.TYPE_INT_RGB);
        // 在新的画布上生成原图的缩略图
        Graphics2D g = tempImg.createGraphics();
        g.setColor(Color.white);
        g.fillRect(0, 0, old_w, old_h);
        g.drawImage(src, 0, 0, old_w, old_h, Color.white, null);
        g.dispose();
        BufferedImage newImg = new BufferedImage(new_w, new_h,
                BufferedImage.TYPE_INT_RGB);
        newImg.getGraphics().drawImage(
                tempImg.getScaledInstance(new_w, new_h, Image.SCALE_SMOOTH), 0,
                0, null);
        // 调用方法输出图片文件
        outImage(outImgPath, newImg, per);
    }

    /**
     * <b>
     * 指定长或者宽的最大值来压缩图片
     * 推荐使用此方法
     * </b>
     *
     * @param file       :源图片路径
     * @param outImgPath :输出的压缩图片的路径
     * @param maxLength  :长或者宽的最大值
     * @param per        :图片质量
     */
    public static void resize(MultipartFile file
            , String outImgPath
            , int maxLength
            , float per) {
        // 得到图片  
        BufferedImage src = InputImage(file);
        int old_w = src.getWidth();
        // 得到源图宽  
        int old_h = src.getHeight();
        // 得到源图长  
        int new_w = 0;
        // 新图的宽  
        int new_h = 0;
        // 新图的长  
        BufferedImage tempImg = new BufferedImage(old_w
                , old_h
                , BufferedImage.TYPE_INT_RGB);
        Graphics2D g = tempImg.createGraphics();
        g.setColor(Color.white);
        // 从原图上取颜色绘制新图
        g.fillRect(0, 0, old_w, old_h);
        g.drawImage(src, 0, 0, old_w, old_h, Color.white, null);
        g.dispose();
        // 根据图片尺寸压缩比得到新图的尺寸  
        if (old_w > old_h) {
            // 图片要缩放的比例  
            new_w = maxLength;
            new_h = (int) Math.round(old_h * ((float) maxLength / old_w));
        } else {
            new_w = (int) Math.round(old_w * ((float) maxLength / old_h));
            new_h = maxLength;
        }
        BufferedImage newImg = new BufferedImage(new_w, new_h,
                BufferedImage.TYPE_INT_RGB);
        newImg.getGraphics().drawImage(
                tempImg.getScaledInstance(new_w, new_h, Image.SCALE_SMOOTH), 0,
                0, null);
        // 调用方法输出图片文件  
        outImage(outImgPath, newImg, per);
    }

    /**
     * 将图片压缩成指定宽度,高度等比例缩放
     *
     * @param file       :源图片路径
     * @param outImgPath :输出的压缩图片的路径
     * @param width      :长或者宽的最大值
     * @param per        :图片质量
     */
    public static void resizeFixedWidth(MultipartFile file
            , String outImgPath
            , int width
            , float per) {
        // 得到图片
        BufferedImage src = InputImage(file);
        int old_w = src.getWidth();
        // 得到源图宽
        int old_h = src.getHeight();
        // 得到源图长
        int new_w = 0;
        // 新图的宽
        int new_h = 0;
        // 新图的长
        BufferedImage tempImg = new BufferedImage(old_w, old_h,
                BufferedImage.TYPE_INT_RGB);
        Graphics2D g = tempImg.createGraphics();
        g.setColor(Color.white);
        // 从原图上取颜色绘制新图
        g.fillRect(0, 0, old_w, old_h);
        g.drawImage(src, 0, 0, old_w, old_h, Color.white, null);
        g.dispose();
        // 根据图片尺寸压缩比得到新图的尺寸
        if (old_w > old_h) {
            // 图片要缩放的比例
            new_w = width;
            new_h = (int) Math.round(old_h * ((float) width / old_w));
        } else {
            new_w = (int) Math.round(old_w * ((float) width / old_h));
            new_h = width;
        }
        BufferedImage newImg = new BufferedImage(new_w, new_h,
                BufferedImage.TYPE_INT_RGB);
        newImg.getGraphics().drawImage(
                tempImg.getScaledInstance(new_w, new_h, Image.SCALE_SMOOTH), 0,
                0, null);
        // 调用方法输出图片文件
        outImage(outImgPath, newImg, per);
    }

    /**
     * 图片剪切工具方法
     *
     * @param srcfile 源图片
     * @param outfile 剪切之后的图片
     * @param x       剪切顶点 X 坐标
     * @param y       剪切顶点 Y 坐标
     * @param width   剪切区域宽度
     * @param height  剪切区域高度
     * @throws IOException
     */
    public static void cut(File srcfile
            , File outfile
            , int x
            , int y
            , int width
            , int height) throws IOException {
        FileInputStream is = null;
        ImageInputStream iis = null;
        try {
            // 读取图片文件
            is = new FileInputStream(srcfile);

            /*
             * 返回包含所有当前已注册 ImageReader 的 Iterator,这些 ImageReader 声称能够解码指定格式。
             * 参数:formatName - 包含非正式格式名称 .(例如 "jpeg" 或 "tiff")等 。
             */
            Iterator<ImageReader> it = ImageIO.getImageReadersByFormatName("jpg");
            ImageReader reader = it.next();
            // 获取图片流
            iis = ImageIO.createImageInputStream(is);

            /*
             * <p>iis:读取源.true:只向前搜索 </p>.将它标记为 ‘只向前搜索’。
             * 此设置意味着包含在输入源中的图像将只按顺序读取,可能允许 reader 避免缓存包含与以前已经读取的图像关联的数据的那些输入部分。
             */
            reader.setInput(iis, true);

            /*
             * <p>描述如何对流进行解码的类<p>.用于指定如何在输入时从 Java Image I/O
             * 框架的上下文中的流转换一幅图像或一组图像。用于特定图像格式的插件 将从其 ImageReader 实现的
             * getDefaultReadParam 方法中返回 ImageReadParam 的实例。
             */
            ImageReadParam param = reader.getDefaultReadParam();

            /*
             * 图片裁剪区域。Rectangle 指定了坐标空间中的一个区域,通过 Rectangle 对象
             * 的左上顶点的坐标(x,y)、宽度和高度可以定义这个区域。
             */
            Rectangle rect = new Rectangle(x, y, width, height);

            // 提供一个 BufferedImage,将其用作解码像素数据的目标。
            param.setSourceRegion(rect);

            /*
             * 使用所提供的 ImageReadParam 读取通过索引 imageIndex 指定的对象,并将 它作为一个完整的
             * BufferedImage 返回。
             */
            BufferedImage bi = reader.read(0, param);

            // 保存新图片
            ImageIO.write(bi, "jpg", outfile);
        } finally {
            if (is != null) {
                is.close();
            }
            if (iis != null) {
                iis.close();
            }
        }
    }

    /**
     * 将图片文件输出到指定的路径,并可设定压缩质量
     *
     * @param outImgPath 输出图片地址
     * @param newImg     新图片
     * @param per        图片质量:
     *                   一些准则:
     *                   0.75高质量 0.5中等质量 0.25低质量
     *                   参数: 质量 0.0-1.0 设置所需的质量级别。
     */
    private static void outImage(String outImgPath
            , BufferedImage newImg
            , float per) {
        // 判断输出的文件夹路径是否存在,不存在则创建  
        File file = new File(outImgPath);
        if (!file.getParentFile().exists()) {
            file.getParentFile().mkdirs();
        }
        // 输出到文件流
        FileOutputStream fos = null;
        try {
            fos = new FileOutputStream(outImgPath);
            JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(fos);
            JPEGEncodeParam jep = JPEGCodec.getDefaultJPEGEncodeParam(newImg);
            // 压缩质量  
            jep.setQuality(per, true);
            encoder.encode(newImg, jep);
            fos.close();
        } catch (Exception e) {
            log.error("outImage: 关闭流失败" + e);
        } finally {
            if (fos != null) {
                try {
                    fos.close();
                } catch (IOException e) {
                    log.error("outImage: 关闭流失败" + e);
                }
            }
        }
    }


    /**
     * 将图片按照指定的图片质量压缩
     *
     * @param file       :图片
     * @param outImgPath :输出的压缩图片的路径
     * @param per        :图片质量压缩
     */
    public static void resize(MultipartFile file
            , String outImgPath
            , float per) {
        // 得到图片
        BufferedImage src = InputImage(file);
        int old_w = src.getWidth();
        // 得到源图宽
        int old_h = src.getHeight();
        // 得到源图长
        // 根据原图的大小生成空白画布
        BufferedImage tempImg = new BufferedImage(old_w, old_h,
                BufferedImage.TYPE_INT_RGB);
        // 在新的画布上生成原图的缩略图
        Graphics2D g = tempImg.createGraphics();
        g.setColor(Color.white);
        g.fillRect(0, 0, old_w, old_h);
        g.drawImage(src, 0, 0, old_w, old_h, Color.white, null);
        g.dispose();
        BufferedImage newImg = new BufferedImage(old_w, old_h,
                BufferedImage.TYPE_INT_RGB);
        newImg.getGraphics().drawImage(
                tempImg.getScaledInstance(old_w, old_h, Image.SCALE_SMOOTH), 0,
                0, null);
        // 调用方法输出图片文件
        outImage(outImgPath, newImg, per);
    }

    /**
     * 获取远程网络图片压缩并上传
     *
     * @param imageURL   远程图片的url
     * @param outImgPath 输出图片地址
     * @param per        图片质量
     */
    public static void getImgByUrl(String imageURL, String outImgPath, float per) {
        URL url = null;
        InputStream is = null;
        BufferedImage src = null;
        try {
            url = new URL(imageURL);
            is = url.openStream();
            src = ImageIO.read(is);
            // 压缩
            int old_w = src.getWidth();
            // 得到源图宽
            int old_h = src.getHeight();
            // 得到源图长
            // 根据原图的大小生成空白画布
            BufferedImage tempImg = new BufferedImage(old_w, old_h,
                    BufferedImage.TYPE_INT_RGB);
            // 在新的画布上生成原图的缩略图
            Graphics2D g = tempImg.createGraphics();
            g.setColor(Color.white);
            g.fillRect(0, 0, old_w, old_h);
            g.drawImage(src, 0, 0, old_w, old_h, Color.white, null);
            g.dispose();
            BufferedImage newImg = new BufferedImage(old_w, old_h,
                    BufferedImage.TYPE_INT_RGB);
            newImg.getGraphics().drawImage(
                    tempImg.getScaledInstance(old_w, old_h, Image.SCALE_SMOOTH), 0,
                    0, null);
            // 调用方法输出图片文件
            outImage(outImgPath, newImg, 0.75F);
        } catch (MalformedURLException e) {
            log.error("getRemoteBufferedImage: " + imageURL + ",无效:" + e);
        } catch (IOException e) {
            log.error("getRemoteBufferedImage: " + imageURL + ",读取失败:" + e);
        } finally {
            try {
                if (is != null) {
                    is.close();
                }
            } catch (IOException e) {
                log.error("getRemoteBufferedImage: " + imageURL + ",流关闭异常:" + e);
            }
        }
    }
}

如果使用以上的工具类打包出现以下错误:

JDK1.8 编译时提示 程序包com.sun.image.codec.jpeg不存在

解决办法:

在pom文件中build -> plugins 中加入以下配置就可以了



 <plugin>
     <artifactId>maven-compiler-plugin</artifactId>
        <configuration>
            <encoding>UTF-8</encoding>
              <compilerArguments>
              <verbose />
<bootclasspath>${java.home}\lib\rt.jar${path.separator}${java.home}\lib\jce.jar${path.separator}</bootclasspath>
              </compilerArguments>
         </configuration>
</plugin>
相关文章
|
2月前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
175 57
|
29天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
2月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
79 8
|
2月前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
91 17
|
2月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
2月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
149 4
|
2月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
98 2
|
2月前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
95 4
|
2月前
|
Java 数据格式 索引
使用 Java 字节码工具检查类文件完整性的原理是什么
Java字节码工具通过解析和分析类文件的字节码,检查其结构和内容是否符合Java虚拟机规范,确保类文件的完整性和合法性,防止恶意代码或损坏的类文件影响程序运行。
64 5
|
2月前
|
Java API Maven
如何使用 Java 字节码工具检查类文件的完整性
本文介绍如何利用Java字节码工具来检测类文件的完整性和有效性,确保类文件未被篡改或损坏,适用于开发和维护阶段的代码质量控制。
144 5