写给大忙人看的 – Java中图片压缩上传至MinIO服务器(4)[通俗易懂]

写给大忙人看的 – Java中图片压缩上传至MinIO服务器(4)[通俗易懂]之前文章已经介绍了MinIO的环境搭建,已经对文件的上传下载方法,本篇文章一起与大家来学习图片压缩上传的方法1、背景最近客户总抱怨APP中图片显示较慢,升级服务器带宽又没有多的预算,查看原因,是因为现在大家都是用的智能手机拍照,拍出来的照片小则2-3M,大则十几M,所以导致图片显示较慢。思考再三,决定将图片进行压缩再上传图片服务器来解决图片显示慢的问题

大家好,又见面了,我是你们的朋友全栈君。

之前文章已经介绍了 MinIO 的环境搭建,已经对文件的上传下载方法,本篇文章一起与大家来学习图片压缩上传的方法

1、背景

最近客户总抱怨 APP 中图片显示较慢, 升级服务器带宽又没有多的预算。查看原因,是因为现在大家都是用的智能手机拍照,拍出来的照片小则 2-3 M,大则十几 M,所以导致图片显示较慢。思考再三,决定将图片进行压缩再上传图片服务器来解决图片显示慢的问题

2、开发前戏

1、引入 maven 依赖

<!-- 图片压缩 -->
<dependency>
    <groupId>net.coobird</groupId>
    <artifactId>thumbnailator</artifactId>
    <version>0.4.8</version>
</dependency>

本次我们选择了使用 thumbnailator 来作为压缩的工具

2、thumbnailator 简介

  • Thumbnailator 是一个用来生成图像缩略图的 Java 类库,通过很简单的代码即可生成图片缩略图,也可直接对一整个目录的图片生成缩略图
  • 支持图片缩放,区域裁剪,水印,旋转,保持比例

3、压缩前戏

  • 判断是否是图片方法
/** * 判断文件是否为图片 */
public boolean isPicture(String imgName) { 
   
    boolean flag = false;
    if (StringUtils.isBlank(imgName)) { 
   
        return false;
    }
    String[] arr = { 
   "bmp", "dib", "gif", "jfif", "jpe", "jpeg", "jpg", "png", "tif", "tiff", "ico"};
    for (String item : arr) { 
   
        if (item.equals(imgName)) { 
   
            flag = true;
            break;
        }
    }
    return flag;
}

3、压缩上传

/** * 上传文件 * * @param file 文件 * @return */
public JSONObject uploadFile(MultipartFile file) throws Exception { 
   
    JSONObject res = new JSONObject();
    res.put("code", 500);
    // 判断上传文件是否为空
    if (null == file || 0 == file.getSize()) { 
   
        res.put("msg", "上传文件不能为空");
        return res;
    }
    // 判断存储桶是否存在
    if (!client.bucketExists("test")) { 
   
        client.makeBucket("test");
    }
    // 拿到文件后缀名,例如:png
    String suffix = file.getOriginalFilename().substring(file.getOriginalFilename().lastIndexOf(".") + 1);
    // UUID 作为文件名
    String uuid = String.valueOf(UUID.randomUUID());
    // 新的文件名
    String fileName = DateUtils.getYyyymmdd() + "/" + uuid + "." + suffix;
    /** * 判断是否是图片 * 判断是否超过了 100K */
    if (isPicture(suffix) && (1024 * 1024 * 0.1) <= file.getSize()) { 
   
        // 在项目根目录下的 upload 目录中生成临时文件
        File newFile = new File(ClassUtils.getDefaultClassLoader().getResource("upload").getPath() + uuid + "." + suffix);
        // 小于 1M 的
        if ((1024 * 1024 * 0.1) <= file.getSize() && file.getSize() <= (1024 * 1024)) { 
   
            Thumbnails.of(file.getInputStream()).scale(1f).outputQuality(0.3f).toFile(newFile);
        }
        // 1 - 2M 的
        else if ((1024 * 1024) < file.getSize() && file.getSize() <= (1024 * 1024 * 2)) { 
   
            Thumbnails.of(file.getInputStream()).scale(1f).outputQuality(0.2f).toFile(newFile);
        }
        // 2M 以上的
        else if ((1024 * 1024 * 2) < file.getSize()) { 
   
            Thumbnails.of(file.getInputStream()).scale(1f).outputQuality(0.1f).toFile(newFile);
        }
        // 获取输入流
        FileInputStream input = new FileInputStream(newFile);
        // 转为 MultipartFile
        MultipartFile multipartFile = new MockMultipartFile("file", newFile.getName(), "text/plain", input);
        // 开始上传
        client.putObject("test", fileName, multipartFile.getInputStream(), file.getContentType());
        // 删除临时文件
        newFile.delete();
        // 返回状态以及图片路径
        res.put("code", 200);
        res.put("msg", "上传成功");
        res.put("url", minioProp.getEndpoint() + "/" + "test" + "/" + fileName);
    }
    // 不需要压缩,直接上传
    else { 
   
        // 开始上传
        client.putObject("test", fileName, file.getInputStream(), file.getContentType());
        // 返回状态以及图片路径
        res.put("code", 200);
        res.put("msg", "上传成功");
        res.put("url", minioProp.getEndpoint() + "/" + "test" + "/" + fileName);
    }
    return res;
}
  • 这里我们判断了当文件为图片的时候,且当它大小超过了 (1024 * 1024 * 0.1),约为 100K 的时候,才进行压缩
  • 我们首先在根目录下的 upload 目录中创建了一个临时文件 newFile
  • Thumbnails.of(file.getInputStream()).scale(1f).outputQuality(0.3f).toFile(newFile);将压缩后的文件输出到临时文件中
  • 然后将 FileInputStream 转为 MultipartFile 上传
  • 最后删除临时文件 newFile.delete();
  • 完成图片压缩上传

4、测试

  • 原图 706K

原图

  • 压缩后 120K

压缩后

5、总结

  • 综合以上代码,可以看出 Thumbnails 对图片的处理是很方便的,且代码量也非常少
  • 通过测试,可以看出压缩后的图片质量也很高
  • thumbnailator 对图片的处理支持全面,缩放,裁剪等

如您在阅读中发现不足,欢迎留言!!!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/150420.html原文链接:https://javaforall.net

(0)
上一篇 2022年6月18日 上午9:46
下一篇 2022年6月18日 上午10:00


相关推荐

  • SpringBoot2 缓存之王caffeine

    SpringBoot2 缓存之王caffeinedependency groupId com github ben manes caffeine groupId artifactId caffeine artifactId version 2 9 0 version dependency 顺便写了个工具类配合 SpringBoot 使用 p

    2026年3月20日
    2
  • 螺旋遍历数组

    螺旋遍历数组假设有一个 5×4 的二维矩阵 现需要按照顺时针方向将其螺旋遍历 返回数组中的元素 若二维数组如上所述 则其顺时针螺旋遍历的结果为 1 2 3 4 5 10 15 20 19 18 17 16 11 6 7 8 9 14 13 12 其实 二维数组的元素排列很像汉字的 回 回 字由一个大矩形包裹着一个小矩形 而二维数组也可以看成是外围的若干元素包裹着内部的元素 由外向内层层嵌套起来 那么我们完全可以按照一定的顺序从外向内一层一层遍历 就等同于对二维数组进行螺旋遍历了 我们可以把二

    2026年3月19日
    3
  • 文本挖掘工具的介绍[通俗易懂]

    文本挖掘工具的介绍[通俗易懂]1、商业文本挖掘的工具2、开源的数据挖掘工具ROSTCM确实是一个很好用的工具。主要用于写论文,真的很好用。LingPipe主要用于自然语言的处理:主题分类(TopClassification)命名实体识别(NamedEntityRecognition)词性标注(Part-ofSpeechTagging)句题检测(Sen

    2022年6月16日
    37
  • visdom总结[通俗易懂]

    visdom总结[通俗易懂]1叫做env=environment2叫做win=window

    2022年6月17日
    47
  • 简单聊聊C/C++中的左值和右值

    简单聊聊C/C++中的左值和右值为什么标题要写成简单聊聊 而不是写成什么 C 中左值与右值详解 或者现在很流行的 惊了 看了这一篇左值与右值讲解 他吊打了面试官 其实带有详解这个词是需要勇气的 最起码要融会贯通之后才敢这么说吧 本来是学习右值引用的 结果涉及到了左值和右值 然后去了解他们历史发现也是有些混乱 操作中又经常涉及到运算符优先级 真是越学越乱了

    2026年3月19日
    2
  • windows安装设置_ffmpeg使用方法

    windows安装设置_ffmpeg使用方法windows下下载配置ffmpeg

    2025年11月7日
    6

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号