小二是新来的实习生,作为技术 leader,我给他安排了一个非常简单的练手任务,把前端 markdown 编辑器里上传的图片保存到服务器端,结果他真的就把图片直接保存到了服务器上,这下可把我气坏了,就不能搞个对象存储服务,比如说 OSS、MinIO?

他理直气壮地反驳道:“谁让你不讲清楚,我去找老板把你开掉!”我瞬间就怂了,说,“来来来,我手把手教你怎么把图片保存到 OSS 上,好不好?”

“不用了,还是我来教你吧。”小二非常自信,下面是他在 Spring Boot 应用中整合 OSS 做的记录。

特此声明:阿里云 OSS 的产品负责人看到后请自觉过来结算下推广费用(狗头)。真心讲:Spring Boot+OSS 在实际开发中挺常见的。

一、开通 OSS

OSS 也就是 Object Storage Service,是阿里云提供的一套对象存储服务,国内的竞品还有七牛云的 Kodo和腾讯云的COS。

第一步,登录阿里云官网,搜索“OSS”关键字,进入 OSS 产品页。

第二步,如果是 OSS 新用户的话,可以享受 6 个月的新人专享优惠价,不过续费的时候还是会肉疼。

第三步,进入 OSS 管理控制台,点击「Bucket 列表」,点击「创建 Bucket」。

Bucket 的词面意思是桶,这里指存储空间,就是用于存储对象的容器。注意读写权限为“公共读”,也就是允许互联网用户访问云空间上的图片。

第四步,点击「确定」就算是开通成功了。

二、整合 OSS

第一步,在 pom.xml 文件中添加 OSS 的依赖。

  1. <!-- 阿里云 OSS -->
  2. <dependency>
  3. <groupId>com.aliyun.oss</groupId>
  4. <artifactId>aliyun-sdk-oss</artifactId>
  5. <version>3.10.2</version>
  6. </dependency>

第二步,在 application.yml 文件中添加 OSS 配置项。

  1. aliyun:
  2. oss:
  3. # oss对外服务的访问域名
  4. endpoint: oss-cn-beijing.aliyuncs.com
  5. # 访问身份验证中用到用户标识
  6. accessKeyId: LTAI5
  7. # 用户用于加密签名字符串和oss用来验证签名字符串的密钥
  8. accessKeySecret: RYN
  9. # oss的存储空间
  10. bucketName: itwanger-oss1
  11. # 上传文件大小(M)
  12. maxSize: 3
  13. # 上传文件夹路径前缀
  14. dir:
  15. prefix: codingmore/images/

第三步,新增 OssClientConfig.java 配置类,主要就是通过 @Value 注解从配置文件中获取配置项,然后创建 OSSClient。

  1. @Configuration
  2. public class OssClientConfig {
  3. @Value("${aliyun.oss.endpoint}")
  4. String endpoint ;
  5. @Value("${aliyun.oss.accessKeyId}")
  6. String accessKeyId ;
  7. @Value("${aliyun.oss.accessKeySecret}")
  8. String accessKeySecret;
  9. @Bean
  10. public OSSClient createOssClient() {
  11. return (OSSClient)new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);
  12. }
  13. }

第四步,新增文件上传接口 OssController.java,参数为 MultipartFile。

  1. @Controller
  2. @Api(tags = "上传")
  3. @RequestMapping("/ossController")
  4. public class OssController {
  5. @Autowired
  6. private IOssService ossService;
  7. @RequestMapping(value = "/upload",method=RequestMethod.POST)
  8. @ResponseBody
  9. @ApiOperation("上传")
  10. public ResultObject<String> upload(@RequestParam("file") MultipartFile file, HttpServletRequest req) {
  11. return ResultObject.success(ossService.upload(file));
  12. }
  13. }

第五步,新增 Service,将文件上传到 OSS,并返回文件保存路径。

  1. @Service
  2. public class OssServiceImpl implements IOssService{
  3. @Value("${aliyun.oss.maxSize}")
  4. private int maxSize;
  5. @Value("${aliyun.oss.bucketName}")
  6. private String bucketName;
  7. @Value("${aliyun.oss.dir.prefix}")
  8. private String dirPrefix;
  9. @Autowired
  10. private OSSClient ossClient;
  11. @Override
  12. public String upload(MultipartFile file) {
  13. try {
  14. return upload(file.getInputStream(), file.getOriginalFilename());
  15. } catch (IOException e) {
  16. LOGGER.error(e.getMessage());
  17. }
  18. return null;
  19. }
  20. @Override
  21. public String upload(InputStream inputStream,String name) {
  22. String objectName = getBucketName(name);
  23. // 创建PutObject请求。
  24. ossClient.putObject(bucketName, objectName, inputStream);
  25. return formatPath(objectName);
  26. }
  27. private String getBucketName(String url){
  28. String ext = "";
  29. for(String extItem:imageExtension){
  30. if(url.indexOf(extItem) != -1){
  31. ext = extItem;
  32. break;
  33. }
  34. }
  35. return dirPrefix+ DateUtil.today()+"/"+ IdUtil.randomUUID()+ext;
  36. }
  37. private String formatPath(String objectName){
  38. return "https://" +bucketName+"."+ ossClient.getEndpoint().getHost() + "/" + objectName;
  39. }
  40. }

第六步,打开 Apipost,测试 OSS 上传接口,注意参数选择文件,点击发送后可以看到服务器端返回的图片链接。

第七步,进入阿里云 OSS 后台管理,可以确认图片确实已经上传成功。

三、拉取前端代码来测试 OSS 上传接口

codingmore-admin-web 是编程喵(Codingmore)的前端管理项目,可以通过下面的地址拉取到本地。

https://github.com/itwanger/codingmore-admin-web

执行 yarn run dev 命令后就可以启动 Web 管理端了,进入到文章编辑页面,选择一张图片进行上传,可以确认图片是可以正常从前端上传到服务器端,服务器端再上传到 OSS,之后再返回前端图片访问链接的。

四、利用 OSS 进行自动转链

第一步,在 PostsServiceImpl.java 中添加图片转链的方法,主要利用正则表达式找出文章内容中的外链,然后将外链的图片上传到 OSS,然后再替换掉原来的外链图片。

  1. // 匹配图片的 markdown 语法
  2. // ![](hhhx.png)
  3. // ![xx](hhhx.png?ax)
  4. public static final String IMG_PATTERN = "\\!\\[.*\\]\\((.*)\\)";
  5. private void handleContentImg(Posts posts) {
  6. String content = posts.getPostContent();
  7. Pattern p = Pattern.compile(IMG_PATTERN, Pattern.CASE_INSENSITIVE);
  8. Matcher m = p.matcher(content);
  9. Map<String, Future<String>> map = new HashMap<>();
  10. while (m.find()) {
  11. String imageTag = m.group();
  12. LOGGER.info("使用分组进行替换{}", imageTag);
  13. String imageUrl = imageTag.substring(imageTag.indexOf("(") + 1, imageTag.indexOf(")"));
  14. // 确认是本站链接,不处理
  15. if (imageUrl.indexOf(iOssService.getEndPoint()) != -1) {
  16. continue;
  17. }
  18. // 通过线程池将图片上传到 OSS
  19. Future<String> future = ossUploadImageExecutor.submit(() -> {
  20. return iOssService.upload(imageUrl);
  21. });
  22. map.put(imageUrl, future);
  23. }
  24. for (String oldUrl : map.keySet()) {
  25. Future<String> future = map.get(oldUrl);
  26. try {
  27. String imageUrl = future.get();
  28. content = content.replace(oldUrl, imageUrl);
  29. } catch (InterruptedException | ExecutionException e) {
  30. LOGGER.error("获取图片链接出错{}", e.getMessage());
  31. }
  32. }
  33. posts.setPostContent(content);
  34. }

第二步,在 OssServiceImpl.java 中添加根据外链地址上传图片到 OSS 的方法。

  1. public String upload(String url) {
  2. String objectName = getFileName(url);
  3. try (InputStream inputStream = new URL(url).openStream()) {
  4. ossClient.putObject(bucketName, objectName, inputStream);
  5. } catch (IOException e) {
  6. LOGGER.error(e.getMessage());
  7. }
  8. return formatOSSPath(objectName);
  9. }

第三步,通过 Web 管理端来测试外链是否转链成功。先找两张外链的图片,可以看到 markdown 在预览的时候就不显示。

然后我们点击发布,可以看到两张图片都正常显示了,因为转成了 OSS 的图片访问地址。

五、小结

综上来看,实习生小二在 Spring Boot 中整合 OSS 的代码还是挺靠谱的。也许 OSS+CDN 才是图床的最好解决方案,不过阿里云的 HTTPS CDN 在 GitHub 上无法回源导致图片不显示的问题仍然没有得到有效的解决。

需要源码的小伙伴可以直接到编程喵源码路径拉取:

https://github.com/itwanger/coding-more


本篇已收录至 GitHub 上星标 1.8k+ star 的开源专栏《Java 程序员进阶之路》,据说每一个优秀的 Java 程序员都喜欢她,风趣幽默、通俗易懂。内容包括 Java 基础、Java 并发编程、Java 虚拟机、Java 企业级开发、Java 面试等核心知识点。学 Java,就认准 Java 程序员进阶之路

https://github.com/itwanger/toBeBetterJavaer

star 了这个仓库就等于你拥有了成为了一名优秀 Java 工程师的潜力。也可以戳下面的链接跳转到《Java 程序员进阶之路》的官网网址,开始愉快的学习之旅吧。

https://tobebetterjavaer.com/

没有什么使我停留——除了目的,纵然岸旁有玫瑰、有绿荫、有宁静的港湾,我是不系之舟

保姆级SpringBoot+Vue图片上传到阿里云OSS教程的更多相关文章

  1. C# .net Ueditor实现图片上传到阿里云OSS 对象存储

    在学习的时候,项目中需要实现在Ueditor编辑器中将图片上传到云储存中,老师演示的是上传到又拍云存储,既然看了一遍,直接照搬不算本事,咱们可以依葫芦画瓢自己来动手玩玩其它的云存储服务. 现在云计算产 ...

  2. egg-multipart + el-upload 实现带参图片上传至阿里云OSS

    egg-multipart有两种模式:file和stream el-upload参数传递有两种方式:利用自带参数data和手动添加参数 egg-multipart介绍 一.file 模式下的带参传递 ...

  3. Windows环境下用C#编程将文件上传至阿里云OSS笔记

    Windows环境下用C#编程将文件上传至阿里云OSS笔记 本系列文章由ex_net(张建波)编写,转载请注明出处. http://blog.csdn.net/ex_net/article/detai ...

  4. Java下载https文件上传到阿里云oss服务器

    Java下载https文件上传到阿里云oss服务器 今天做了一个从Https链接中下载音频并且上传到OSS服务器,记录一下希望大家也少走弯路. 一共两个类: 1 .实现自己的证书信任管理器类 /** ...

  5. Linux本地数据上传到阿里云OSS

    这篇文章主要是介绍如何将服务器本地的数据上传到阿里云OSS的指定bucket中,最重要的参考文档是数据迁移单机部署.我第一次上传数据到OSS上时,步骤要比前面的链接中介绍的要麻烦,ossimport工 ...

  6. 备份MySQL数据库并上传到阿里云OSS存储

    1. 环境配置 要将本地文件上传到阿里云oss中, 必须使用阿里云提供的工具 ossutil, 有32位,也有64位的, Linux和Windows都有.具体可以到阿里云官网下载 官网及文档: htt ...

  7. 前端(react)上传到阿里云OSS存储 实例

    需求背景 由于现有的后台管理系统,上传的视频越来越大,加上上传视频较慢,后端小哥提出直接从前端上传视频或者其他文件到阿里云OSS存储. 阿里云OSS 阿里云OSS文档介绍,这里不做过多赘述 安装 原本 ...

  8. springmvc学习笔记--支持文件上传和阿里云OSS API简介

    前言: Web开发中图片上传的功能很常见, 本篇博客来讲述下springmvc如何实现图片上传的功能. 主要讲述依赖包引入, 配置项, 本地存储和云存储方案(阿里云的OSS服务). 铺垫: 文件上传是 ...

  9. ThinkPHP 文件上传到阿里云OSS上(干货)

    参考:http://www.thinkphp.cn/extend/789.html 1.前往阿里云github下载SDK包:https://github.com/aliyun/aliyun-oss-p ...

随机推荐

  1. Shell之sed编辑器

    Shell之sed编辑器 目录 Shell之sed编辑器 一.sed编辑器 1. sed编辑器概述 2. sed编辑器的工作流程 二.sed命令 1. 命令格式 2. 常用选项 3. 常用操作 三.操 ...

  2. MATLAB基础学习(3)——数值数组及运算

    rand('state',s)表示随机产生数的状bai态state,一般情百况du下不用指定状态.rand('state',0)作用在于如果指容定zhi状态,产生dao随机结果就相同了.一般情况下不用 ...

  3. VScode git无法使用,Error: command 'git.push' not found 源代码管理无法使用的问题及解决方法

    正常条件下,只要电脑中安装了Git,VScode就可以直接使用. 在开始界面有下图所示的功能: 在源代码管理栏目中: 如果没能正常工作,就看不到这些功能. 可能在用某些与git相关的功能时,如安装了G ...

  4. Kubernetes:容器资源需求与限制(约束)

    Blog:博客园 个人 A Container is guaranteed to have as much memory as it requests, but is not allowed to u ...

  5. uni-app、Vue3 + ucharts 图表 H5 无法渲染

    文章已收录到 github,欢迎 Watch 和 Star. 简介 从问题定位开始,到给框架(uni-app)提 issue.出解决方案(PR),再到最后的思考,详细记录了整个过程. 前序 当你在业务 ...

  6. 矩阵QR分解

    1 orthonormal 向量与 Orthogonal 矩阵 orthonormal 向量定义为 ,任意向量  相互垂直,且模长为1: 如果将  orthonormal 向量按列组织成矩阵,矩阵为  ...

  7. Solution Set - Stirling 数相关杂题

      <好多题的题解>   「洛谷 P5408」第一类斯特林数·行   根据结论 \[x^{\overline{n}}=\sum_i{n\brack i}x^i, \] 我们只需要求出 \( ...

  8. HashMap(1.7)源码学习

    一. 1.7 和1.8区别 数据结构: 1.7: 数组 + 链表 1.8 : 数组 + 链表 + 红黑树 put: 1.7: 头插法 1.8: 尾插法 hash计算: 1.7 : Objects.ha ...

  9. 已经安装的nginx增加额外配置步骤

    这里以安装第三方ngx_http_google_filter_module模块为例nginx的模块是需要重新编译nginx,而不是像apache一样配置文件引用.so1. 下载第三方扩展模块ngx_h ...

  10. Linux系列——挂载Windows虚拟文件夹到Linux系统

    ​ 在windows操作系统上安装多台Linux虚拟机,需要方便的在windows系统和虚拟机上的Linux系统进行文件拷贝. 需要用到共享虚拟文件夹技术,将windows文件夹挂载到linux系统中 ...