后端开发中的文件上传的实现

1. 配置 application.yml

application.yml 中配置阿里云 OSS 的相关信息:

bash 复制代码
aliyun:
  oss:
    endpoint: "<您的OSS地域节点>" # 示例: oss-cn-hangzhou.aliyuncs.com
    accessKeyId: "<您的AccessKeyId>"
    accessKeySecret: "<您的AccessKeySecret>"
    bucketName: "<您的Bucket名称>"

2. 创建 AliOssConfig 配置类

在后端创建一个配置类用于初始化阿里云 OSS 客户端实例。

java 复制代码
package com.sky.properties;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

@Component
@ConfigurationProperties(prefix = "sky.alioss")
@Data
public class AliOssProperties {

    private String endpoint;
    private String accessKeyId;
    private String accessKeySecret;
    private String bucketName;

}
  • @ConfigurationProperties 注解通过 前缀匹配 ,将配置文件(如 application.ymlapplication.properties)中的属性自动绑定到类的字段。
  • 需要使用 @Component 或其他形式将类注入到 Spring 容器中。
java 复制代码
package com.sky.config;

import com.sky.properties.AliOssProperties;
import com.sky.utils.AliOssUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * 配置类,用于创建AliOssUtil
 */
@Configuration
@Slf4j
public class OssConfiguration {

    @Bean
    @ConditionalOnMissingBean
    public AliOssUtil aliOssUtil(AliOssProperties aliOssProperties){
        log.info("开始创建阿里云文件上传工具类对象{}:",aliOssProperties);
        return  new AliOssUtil(aliOssProperties.getEndpoint(),
                aliOssProperties.getAccessKeyId(),
                aliOssProperties.getAccessKeySecret(),
                aliOssProperties.getBucketName());
    }
}
  • 动态创建 AliOssUtil 工具类对象

    使用配置类方式来初始化 AliOssUtil 工具类,并将配置文件中的属性通过 AliOssProperties 传递给工具类。

  • 自动注入 AliOssProperties

    方法参数 AliOssProperties aliOssProperties 会自动从 Spring 容器中注入,因为 AliOssProperties 类已经被标注了 @Component@ConfigurationProperties

  • 便于工具类的全局复用

    通过将 AliOssUtil 注册为 Bean,其他任何需要用到文件上传的地方都可以通过自动注入(@Autowired)的方式获取实例。

3. 创建 AliOssUtil 工具类

封装一个工具类来处理文件上传的核心逻辑。直接找搬就好了

java 复制代码
package com.sky.utils;

import com.aliyun.oss.ClientException;
import com.aliyun.oss.OSS;
import com.aliyun.oss.OSSClientBuilder;
import com.aliyun.oss.OSSException;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import java.io.ByteArrayInputStream;

@Data
@AllArgsConstructor
@Slf4j
public class AliOssUtil {

    private String endpoint;
    private String accessKeyId;
    private String accessKeySecret;
    private String bucketName;

    /**
     * 文件上传
     *
     * @param bytes
     * @param objectName
     * @return
     */
    public String upload(byte[] bytes, String objectName) {

        // 创建OSSClient实例。
        OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);

        try {
            // 创建PutObject请求。
            ossClient.putObject(bucketName, objectName, new ByteArrayInputStream(bytes));
        } catch (OSSException oe) {
            System.out.println("Caught an OSSException, which means your request made it to OSS, "
                    + "but was rejected with an error response for some reason.");
            System.out.println("Error Message:" + oe.getErrorMessage());
            System.out.println("Error Code:" + oe.getErrorCode());
            System.out.println("Request ID:" + oe.getRequestId());
            System.out.println("Host ID:" + oe.getHostId());
        } catch (ClientException ce) {
            System.out.println("Caught an ClientException, which means the client encountered "
                    + "a serious internal problem while trying to communicate with OSS, "
                    + "such as not being able to access the network.");
            System.out.println("Error Message:" + ce.getMessage());
        } finally {
            if (ossClient != null) {
                ossClient.shutdown();
            }
        }

        //文件访问路径规则 https://BucketName.Endpoint/ObjectName
        StringBuilder stringBuilder = new StringBuilder("https://");
        stringBuilder
                .append(bucketName)
                .append(".")
                .append(endpoint)
                .append("/")
                .append(objectName);

        log.info("文件上传到:{}", stringBuilder.toString());

        return stringBuilder.toString();
    }
}

4. 创建 FileUploadController 控制器

编写 Controller 层接口,供前端调用。

java 复制代码
package com.sky.controller.admin;

import com.sky.constant.MessageConstant;
import com.sky.result.Result;
import com.sky.utils.AliOssUtil;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;

import java.io.IOException;
import java.util.UUID;

@RestController
@RequestMapping("/admin/common")
@Api(tags = "通用接口")
@Slf4j
public class CommonController {
    @Autowired
    private AliOssUtil  aliOssUtil;
    /**
     * 文件上传
     * @param file
     * @return
     */
    @PostMapping("/upload")
    @ApiOperation("文件上传")
    public Result<String> upload(MultipartFile file){
        log.info("文件上传 :{}",file);
        //
        try {
            // 防止文件重命到时文件覆盖,需要使用UUID,将文件重命名
            // 原始文件名
            String originalFilename = file.getOriginalFilename();
            //截取原始文件的后缀名
            String extention = originalFilename.substring(originalFilename.lastIndexOf("."));
            // 构造新文件名称
            String objectName = UUID.randomUUID().toString() + extention;
            // 文件的请求路径
            String filePath = aliOssUtil.upload(file.getBytes(), objectName);
            return Result.success(filePath);
        } catch (IOException e) {
            log.error("文件上传失败 :" ,e);
        }
         return Result.error(MessageConstant.UPLOAD_FAILED);
    }
}
相关推荐
该用户已不存在14 分钟前
Vibe Coding 入门指南:从想法到产品的完整路径
前端·人工智能·后端
野生技术架构师14 分钟前
牛客网Java 高频面试题总结(2025最新版)
java·开发语言·面试
申阳19 分钟前
Day 3:01. 基于Nuxt开发个人呢博客项目-初始化项目
前端·后端·程序员
铁锹少年20 分钟前
当多进程遇上异步:一次 Celery 与 Async SQLAlchemy 的边界冲突
分布式·后端·python·架构·fastapi
纪莫23 分钟前
技术面:SpringBoot(springboot的类加载和传统的双亲委派有什么区别、如何按顺序实例化Bean)
java·spring·java面试⑧股
曾经的三心草35 分钟前
springcloud二-Seata3- Seata各事务模式
后端·spring·spring cloud
王中阳Go39 分钟前
又整理了一场真实Golang面试复盘!全是高频坑+加分话术,面试遇到直接抄
后端·面试·go
kyle~41 分钟前
CPU调度---协程
java·linux·服务器·数据库·c++20
JavaGuide43 分钟前
今年小红书后端开出了炸裂的薪资!
后端·面试
会飞的小蛮猪1 小时前
Skywalking运维之路(Skywalking服务搭建)
java·运维·监控