SpringBoot整合Minio及阿里云OSS(配置文件无缝切换)

news2024/9/24 7:14:19

SpringBoot整合Minio及阿里云OSS

文章目录

  • SpringBoot整合Minio及阿里云OSS
    • 1.Minio安装测试
      • 1.Docker安装
        • 启动容器
      • 2.创建bucket
      • 3.上传文件
        • 修改权限
    • 2.SpringBoot整合Minio及阿里云OSS
      • 1.公共部分抽取
      • 2.Minio配置整合
        • 1.添加pom依赖
        • 2.添加配置文件
        • 3.操作接口实现
      • 3.阿里云OSS配置整合
        • 1.pom依赖
        • 2.添加配置文件
        • 3.操作接口实现
      • 4.测试

1.Minio安装测试

MinIO是一个对象存储解决方案,它提供了与Amazon Web Services S3兼容的API,并支持所有核心S3功能。 MinIO有能力在任何地方部署 - 公有云或私有云,裸金属基础设施,编排环境,以及边缘基础设施。

文档地址:https://www.minio.org.cn/docs/minio/linux/developers/java/API.html#

1.Docker安装

拉取对应的镜像

docker pull minio/minio

创建挂载目录

mkdir -p /dockerData/minio/data
mkdir -p /dockerData/minio/config
启动容器

然后我们启动我们的容器,后面有个目录,就是我们需要挂载的硬盘目录

docker run --privileged -it -p 9000:9000 --name minio \
-e "MINIO_ACCESS_KEY=moshangshang2024" \
--privileged=true \
-e "MINIO_SECRET_KEY=moshangshang2024" \
-v /dockerData/minio/data:/data \
-v /dockerData/minio/config:/root/.minio \
minio/minio server /data

最新版本的minio启动使用这条语句

其中修改了MINIO_ROOT_USERMINIO_ROOT_PASSWORD名称,增加了web控制台端口,密码长度需大于8位

docker run  --privileged -it \
--name minio \
-p 9000:9000  \
-p 9090:9090  \
-d --restart=always \
-e "MINIO_ROOT_USER=moshangshang2024" \
-e "MINIO_ROOT_PASSWORD=xxxxxxxx" \
-v /dockerData/minio/data:/data \
-v /dockerData/minio/config:/root/.minio \
minio/minio server  /data --console-address ":9090" --address ":9000"

我们只需要访问上面提到的ip地址

http://192.168.1.101:9000

输入刚刚配置的账号moshangshang2024和密码 即可进入
在这里插入图片描述

2.创建bucket

我们首先需要创建一个桶,可以当成是一个目录,选择 create bucket进行创建

在这里插入图片描述

3.上传文件

然后我们选中我们的桶,选择 upload 进行文件上传

在这里插入图片描述

在这里插入图片描述

修改权限

如果要使用SDK,比如Java客户端来操作我们的minio的话,那么我们还需要修改一下我们的bucket权限

在这里插入图片描述
在这里插入图片描述

然后就可以通过http://ip:9000/存储桶名/文件名访问文件

2.SpringBoot整合Minio及阿里云OSS

1.公共部分抽取

1.添加自定义yml配置

#对象存储
oss:
  #对象存储切换配置
  type: minio
  minio:
    endpoint: http://192.168.1.102:9000
    accessKey: root
    secretKey: root
    bucketImageName: test

  #阿里云对象存储的配置信息
  aliyun:
    accessKey: xxxxx
    accessSecret: xxxxx
    endpoint: oss-cn-hangzhou.aliyuncs.com
    bucketImageName: test

2.公共操作方法接口

package com.li.test.minio;

import org.springframework.web.multipart.MultipartFile;

import javax.servlet.http.HttpServletResponse;
import java.util.List;

public interface OSSOperation {

    /**
     * 获取默认bucketName
     *
     * @return 返回 名称
     */
    public String getBucketName();

    /**
     * 校验bucket是否存在
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果存储桶存在,则为 True。
     */
    public boolean checkBucketExist(String bucketName);

    /**
     * 列出所有存储桶的存储桶信息
     */
    public List<String> listBuckets();

    /**
     * 创建一个存储桶
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果执行成功,则为 True。
     */
    public boolean makeBucket(String bucketName);


    /**
     * 删除一个空的存储桶
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果执行成功,则为 True。
     */
    public boolean removeBucket(String bucketName);


    /**
     * 文件上传
     *
     * @param data       文件数据
     * @param bucketName 上传的桶名称
     */
    public boolean uploadFile(MultipartFile data, String bucketName);

    /**
     * 文件上传
     *
     * @param fileName   文件名
     * @param bucketName 上传的桶名称
     */
    public void downloadFile( String fileName, String bucketName, HttpServletResponse response);

    /**
     * 文件删除
     *
     * @param fileName   文件名
     * @param bucketName 上传的桶名称
     */
    public boolean removeFile(String fileName, String bucketName);
}

2.Minio配置整合

1.添加pom依赖
      <dependency>
            <groupId>io.minio</groupId>
            <artifactId>minio</artifactId>
            <version>8.5.10</version>
        </dependency>
2.添加配置文件

然后我们需要编写配置文件,用于初始化配置 MinioClient装载到spring容器中

@Data
@ConfigurationProperties(prefix = "oss.minio")
public class MinioOSSProperties {

    private String endpoint;
    private String accessKey;
    private String secretKey;
    private String bucketImageName;

}

/**
 * Minio配置类
 *
 * @author moshangshang
 */
@Configuration
@EnableConfigurationProperties(MinioOSSProperties.class)
@ConditionalOnProperty(prefix = "oss", name = "type",havingValue = "minio", matchIfMissing = true)
public class MinioOSSConfiguration {

    @Resource
    private MinioOSSProperties ossProperties;

    @Bean
    @SneakyThrows
    public MinioClient minioClient() {
        return MinioClient.builder()
                .endpoint(ossProperties.getEndpoint())
                .credentials(ossProperties.getAccessKey(), ossProperties.getSecretKey())
                .build();
    }

}
3.操作接口实现
/**
 * minio操作工具类
 * @author moshangshang
 */
@Slf4j
@Data
@Component
public class MinioUtils {

    @Resource
    private MinioClient minioClient;

    @Resource
    private OssMinioProperties minioProperties;


    /**
     * 校验bucket是否存在
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果存储桶存在,则为 True。
     */
    public boolean checkBucketExist(String bucketName) {
        boolean found = false;
        try {
            found = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build());
            if (found) {
                log.info("{} exists", bucketName);
            } else {
                log.info("{} not exist", bucketName);
            }
        } catch (Exception e) {
            log.info("{} checkBucketExist exception", bucketName,e);
        }
        return found;
    }

    /**
     * 列出所有存储桶的存储桶信息
     */
    public List<Bucket> listBuckets() {
        List<Bucket> buckets = new ArrayList<>();
        try {
            buckets = minioClient.listBuckets();
        } catch (Exception e) {
            log.info("listBuckets exception......",e);
        }
        return buckets;

    }

    /**
     * 创建一个存储桶
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果执行成功,则为 True。
     */
    public boolean makeBucket(String bucketName) {
        boolean found = false;
        try {
            minioClient.makeBucket(
                    MakeBucketArgs.builder()
                            .bucket(bucketName)
                            .build());
            found = true;
        } catch (Exception e) {
            log.info("{} makeBucket exception {}", bucketName,e.getMessage(),e);
        }
        return found;
    }


    /**
     * 删除一个空的存储桶
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果执行成功,则为 True。
     */
    public boolean removeBucket(String bucketName) {
        boolean found = false;
        try {
            minioClient.removeBucket(RemoveBucketArgs.builder().bucket(bucketName).build());
            found = true;
        } catch (Exception e) {
            log.info("{} removeBucket exception", bucketName);
        }
        return found;
    }


    /**
     * 文件上传
     *
     * @param data       文件数据
     * @param bucketName 上传的桶名称
     */
    public boolean uploadFile(MultipartFile data, String bucketName) {
        boolean flag = checkBucketExist(bucketName);
        if (!flag){
            return false;
        }
        String fileName = data.getOriginalFilename();
        InputStream is = null;
        try {
            is = data.getInputStream();
            minioClient.putObject(
                    PutObjectArgs.builder()
                            .bucket(bucketName)
                            .object(fileName)
                            .stream(is, data.getSize(), -1)
                            .contentType(data.getContentType())
                            .build());
            return true;
        } catch (Exception e) {
            log.info("{} upload exception", bucketName,e);
        }
        return false;
    }

    /**
     * 文件上传
     *
     * @param fileName   文件名
     * @param bucketName 上传的桶名称
     */
    public void downloadFile( String fileName, String bucketName, HttpServletResponse response) {
        GetObjectResponse is = null;
        try {
            GetObjectArgs getObjectArgs = GetObjectArgs.builder()
                    .bucket(bucketName)
                    .object(fileName)
                    .build();
            is = minioClient.getObject(getObjectArgs);
            response.setContentType("application/octet-stream");
            response.setCharacterEncoding("utf-8");
            response.setHeader("Content-Disposition", "attachment;fileName=" + URLEncoder.encode(fileName, "utf-8"));
            IoUtil.copy(is, response.getOutputStream());
            log.info("minio downloadFile success, filePath:{}", fileName);
        } catch (Exception e) {
            log.error("minio downloadFile Exception:{}", e.getMessage(), e);
        } finally {
            IoUtil.close(is);
        }
    }

    /**
     * 文件删除
     *
     * @param fileName   文件名
     * @param bucketName 上传的桶名称
     */
    public boolean removeFile(String fileName, String bucketName) {
        boolean flag = checkBucketExist(bucketName);
        if (!flag){
            return false;
        }
        try {
            minioClient.removeObject(
                    RemoveObjectArgs.builder().bucket(bucketName).object(fileName).build());
            return true;
        } catch (Exception e) {
            log.error("minio removeFile Exception:{}", e.getMessage(), e);
        }
        return false;
    }
}

3.阿里云OSS配置整合

1.pom依赖
     <dependency>
            <groupId>com.aliyun.oss</groupId>
            <artifactId>aliyun-sdk-oss</artifactId>
            <version>3.10.2</version>
        </dependency>
2.添加配置文件
@Data
@ConfigurationProperties(prefix = "oss.aliyun")
public class AliYunOSSProperties {

    private String accessKey;

    private String accessSecret;

    private String endpoint;

    private String bucketImageName;

}
/**
 * 阿里云oss配置类
 *
 * @author moshangshang
 */
@Configuration
@EnableConfigurationProperties(AliYunOSSProperties.class)
@ConditionalOnProperty(prefix = "oss", name = "type",havingValue = "aliyun", matchIfMissing = true)
public class AliYunOSSConfiguration {

    @Resource
    private AliYunOSSProperties ossProperties;

    @Bean
    @SneakyThrows
    public OSS ossClient() {
        return new OSSClientBuilder().build(ossProperties.getEndpoint(),ossProperties.getAccessKey(),ossProperties.getAccessSecret());
    }

}
3.操作接口实现
/**
 * 阿里云oss操作工具类
 * @author moshangshang
 */
@Slf4j
@Data
@Component
@ConditionalOnProperty(prefix = "oss", name = "type",havingValue = "aliyun", matchIfMissing = true)
public class AliYunOSSOperation implements OSSOperation {

    @Resource
    private OSS ossClient;

    @Resource
    private AliYunOSSProperties aliYunOSSProperties;


    @Override
    public String getBucketName() {
        return aliYunOSSProperties.getBucketImageName();
    }

    /**
     * 校验bucket是否存在
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果存储桶存在,则为 True。
     */
    public boolean checkBucketExist(String bucketName) {
        boolean found = false;
        try {
            found = ossClient.doesBucketExist(bucketName);
            if (found) {
                log.info("{} exists", bucketName);
            } else {
                log.info("{} not exist", bucketName);
            }
        } catch (Exception e) {
            log.info("{} checkBucketExist exception", bucketName,e);
        }
        return found;
    }

    /**
     * 列出所有存储桶的存储桶信息
     */
    public List<String> listBuckets() {
        List<String> result = new ArrayList<>();
        try {
           List<Bucket> buckets = ossClient.listBuckets();
           result = buckets.stream().map(Bucket::getName).collect(Collectors.toList());
        } catch (Exception e) {
            log.info("listBuckets exception......",e);
        }
        return result;

    }

    /**
     * 创建一个存储桶
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果执行成功,则为 True。
     */
    public boolean makeBucket(String bucketName) {
        boolean found = false;
        try {
            ossClient.createBucket(bucketName);
            found = true;
        } catch (Exception e) {
            log.info("{} makeBucket exception {}", bucketName,e.getMessage(),e);
        }
        return found;
    }


    /**
     * 删除一个空的存储桶
     *
     * @param bucketName 桶名称
     * @return 返回 boolean - 如果执行成功,则为 True。
     */
    public boolean removeBucket(String bucketName) {
        boolean found = false;
        try {
            ossClient.deleteBucket(bucketName);
            found = true;
        } catch (Exception e) {
            log.info("{} removeBucket exception", bucketName);
        }
        return found;
    }


    /**
     * 文件上传
     *
     * @param data       文件数据
     * @param bucketName 上传的桶名称
     */
    public boolean uploadFile(MultipartFile data, String bucketName) {
        String fileName = data.getOriginalFilename();
        InputStream is = null;
        try {
            is = data.getInputStream();
            ossClient.putObject(bucketName,fileName,data.getInputStream());
            return true;
        } catch (Exception e) {
            log.info("{} upload exception", bucketName,e);
        }
        return false;
    }

    /**
     * 文件下载
     *
     * @param fileName   文件名
     * @param bucketName 上传的桶名称
     */
    public void downloadFile( String fileName, String bucketName, HttpServletResponse response) {
        InputStream is = null;
        try {

            is = ossClient.getObject(bucketName,fileName).getObjectContent();
            response.setContentType("application/octet-stream");
            response.setCharacterEncoding("utf-8");
            response.setHeader("Content-Disposition", "attachment;fileName=" + URLEncoder.encode(fileName, "utf-8"));
            IoUtil.copy(is, response.getOutputStream());
            log.info("minio downloadFile success, filePath:{}", fileName);
        } catch (Exception e) {
            log.error("minio downloadFile Exception:{}", e.getMessage(), e);
        } finally {
            IoUtil.close(is);
        }
    }

    /**
     * 文件删除
     *
     * @param fileName   文件名
     * @param bucketName 上传的桶名称
     */
    public boolean removeFile(String fileName, String bucketName) {
        boolean flag = checkBucketExist(bucketName);
        if (!flag){
            return false;
        }
        try {
            ossClient.deleteObject(bucketName,fileName);
            return true;
        } catch (Exception e) {
            log.error("minio removeFile Exception:{}", e.getMessage(), e);
        }
        return false;
    }
}

4.测试

/**
 * minio测试
 * @author moshangshang
 */
@Slf4j
@RestController
public class OSSController {

    @Resource
    private OSSOperation ossOperation;

    
    @PostMapping("/check")
    public boolean checkBucketExist(@RequestParam("bucketName")String bucketName) {
        return ossOperation.checkBucketExist(bucketName);
    }

    @PostMapping("/upload")
    public boolean upload(@RequestParam("data") MultipartFile data) {
        return ossOperation.uploadFile(data, ossOperation.getBucketName());
    }

    @PostMapping("/download")
    public void download(@RequestParam("fileName")String fileName, HttpServletResponse response) {
        ossOperation.downloadFile(fileName, ossOperation.getBucketName(), response);
    }

    @PostMapping("/remove/file")
    public boolean removeFile(@RequestParam("fileName")String fileName) {
        return ossOperation.removeFile(fileName, ossOperation.getBucketName());
    }

    @PostMapping("/remove/bucket")
    public boolean removeBucket(@RequestParam("bucketName")String bucketName) {
        return ossOperation.removeBucket(bucketName);
    }

    @PostMapping("/add/bucket")
    public boolean makeBucket(@RequestParam("bucketName")String bucketName) {
        return ossOperation.makeBucket(bucketName);
    }


    @PostMapping("/bucket/list")
    public List<String> listBuckets() {
        return ossOperation.listBuckets();
    }


}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2110627.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Class4——Esp32|Thonny两种方式同过电脑控制LED灯,路由器与电脑自带热点连接ESP32

上一节我们通过路由器和设备创建了连接&#xff0c;不懂可按上节配置 Class3——Esp32|Thonny——网络连接主机-wifi连接&#xff08;源代码带教程&#xff09;-CSDN博客文章浏览阅读57次。Esp32|Thonny网络连接主机-wifi连接&#xff08;源代码带教程&#xff09;https://blo…

免费开源的低代码表单FormCreate安装教程,支持可视化设计,适配移动端

低代码表单FormCreate 是一个可以通过 JSON 生成具有动态渲染、数据收集、验证和提交功能的表单生成组件。它支持 6 个 UI 框架&#xff0c;适配移动端&#xff0c;并且支持生成任何 Vue 组件。内置 20 种常用表单组件和自定义组件&#xff0c;再复杂的表单都可以轻松搞定 源码…

网页时装购物:Spring Boot框架的创新应用

第2章相关技术 2.1 B/S架构 B/S结构的特点也非常多&#xff0c;例如在很多浏览器中都可以做出信号请求。并且可以适当的减轻用户的工作量&#xff0c;通过对客户端安装或者是配置少量的运行软件就能够逐步减少用户的工作量&#xff0c;这些功能的操作主要是由服务器来进行控制的…

时尚购物革命:Spring Boot技术在网页时装系统中的应用

第1章 绪论 1.1背景及意义 随着社会的快速发展&#xff0c;计算机的影响是全面且深入的。人们生活水平的不断提高&#xff0c;日常生活中人们对时装购物系统方面的要求也在不断提高&#xff0c;喜欢购物的人数更是不断增加&#xff0c;使得时装购物系统的开发成为必需而且紧迫的…

Rspack 1.0 发布了!

文章来源&#xff5c;Rspack Team 项目地址&#xff5c;https://github.com/web-infra-dev/rspack Rspack 是基于 Rust 编写的下一代 JavaScript 打包工具&#xff0c; 兼容 webpack 的 API 和生态&#xff0c;并提供 10 倍于 webpack 的构建性能。 在 18 个月前&#xff0c;我…

深度学习 --- VGG16能让某个指定的feature map激活值最大化图片的可视化(JupyterNotebook实战)

VGG16能让某个指定的feature map激活值最大化图片的可视化 在前面的文章中&#xff0c;我用jupyter notebook分别实现了&#xff0c;预训练好的VGG16模型各层filter权重的可视化和给VGG16输入了一张图像&#xff0c;可视化VGG16各层的feature map。深度学习 --- VGG16卷积核的可…

说一下场外的伦敦银交易的技巧

在很多讨论伦敦银交易技巧的文章中&#xff0c;一上来就介绍各种交易指标、K线信号等等&#xff0c;这种开门见山的方式很直接也很方便&#xff0c;但也容易忽略了一些场外的技巧&#xff0c;下面我们就来讨论一下场外的关于伦敦银交易的技巧。 何为场外的技巧呢&#xff1f;场…

Java进阶13讲__第十讲__精简

字节流 字节输入流&#xff1a;FileInputStream&#xff08;原始流/低级流&#xff09; 字节缓冲流&#xff1a;BufferedInputStream&#xff08;包装流/处理流&#xff09; 参数是"低级流" 字节输入流/缓冲流常用格式 byte[] arr new byte[1024];//字节流 int l…

灯塔:MYSQL笔记(2)函数

函数 是指一段可以直接被另一段程序调用的程序或代码。 字符串函数 SELECT 函数(参数); 数值函数 SELECT 函数(参数); -- 生成一个六位验证码 select lpad(round(rand()*1000000,0) ,6,0)as 验证码; 日期函数 流程函数 总结&#xff1a; 约束&#xff1a; 1. 概述&#xff…

vim 安装与配置教程(详细教程)

vim就是一个功能非常强大的文本编辑器&#xff0c;可以自己DIY的那种 &#xff0c;不但可以写代码 &#xff0c;还可编译 &#xff0c;可以让你手不离键盘的完成鼠标的所有操作。 如果想要了解vim的的发展历史和详细解说&#xff0c;可以自行上网搜索&#xff0c;我主要是记录一…

第T10周:数据增强

>- **&#x1f368; 本文为[&#x1f517;365天深度学习训练营](https://mp.weixin.qq.com/s/0dvHCaOoFnW8SCp3JpzKxg) 中的学习记录博客** >- **&#x1f356; 原作者&#xff1a;[K同学啊](https://mtyjkh.blog.csdn.net/)** 在本教程中&#xff0c;你将学会如何进行数…

【NLP自然语言处理】文本处理的基本方法

目录 &#x1f354;什么是分词 &#x1f354;中文分词工具jieba 2.1 jieba的基本特点 2.2 jieba的功能 2.3 jieba的安装及使用 &#x1f354;什么是命名实体识别 &#x1f354;什么是词性标注 &#x1f354;小结 学习目标 &#x1f340; 了解什么是分词, 词性标注, 命名…

Java笔试面试题AI答之JDBC(3)

文章目录 13. 编写JDBC连Oracle的程序?14. 简述JDBC的主要组件有哪些 &#xff1f;15. JDBC中如何防止SQL注入攻击&#xff1f;1. 使用预处理语句&#xff08;PreparedStatement&#xff09;2. 避免在SQL查询中直接拼接用户输入的数据总结 16. JDBC的脏读是什么&#xff1f;哪…

Windows下Python和PyCharm的应用(一)__第一个测试程序

1、下载Python安装包 直接从官网下载&#xff0c;百度里搜出来的Python下载&#xff0c;很多是别的公司的商业广告&#xff0c;千万要注意&#xff0c;不要乱点进去&#xff0c;免得浪费时间。 从官网下载&#xff0c;链接&#xff1a;Download Python | Python.org 2、安装Pyt…

网络编程day03(网络体系结构、调试命令、TCP/IP对比)

目录 1》网络的体系结构 1> OSI模型 2> TCP/IP模型 3> 常见网络协议 4> DNS域名解析协议 2》 网络调试命令 1> ping&#xff1a;测试网络连通性&#xff08;ICMP&#xff09; 2> netstat 3》Dos &#xff08;拒绝式服务&#xff09;攻击&#xff1f;…

怎么在mathtype中打空格 MathType空格键不能用

MathType是一款数学公式编辑器&#xff0c;可以帮助用户创建复杂的数学公式和方程式。它提供了一个用户友好的界面&#xff0c;使得编辑和排版数学公式变得更加容易和高效。用户可以直接在其界面中输入公式&#xff0c;也可以将已有的公式从其他文档中复制粘贴过来进行编辑。在…

【2024数模国赛赛题思路公开】国赛B题第二套思路丨附可运行代码丨无偿自提

2024年数模国赛B题解题思路 B 题 生产过程中的决策问题 一、问题1解析 问题1的任务是为企业设计一个合理的抽样检测方案&#xff0c;基于少量样本推断整批零配件的次品率&#xff0c;帮助企业决定是否接收供应商提供的这批零配件。具体来说&#xff0c;企业需要依据两个不同…

秋燥拜拜,中秋润起来,酒茶香中秋有“礼”

话说这初秋啊&#xff0c;真是个让人又爱又恨的季节&#xff01; 爱它的秋高气爽&#xff0c;恨它的天干物燥。就像是我们刚刚结束了一个炎热的夏天&#xff0c;身体还没来得及适应&#xff0c;就被秋天的干燥给来了个“突然袭击”。鼻子干、嘴唇干、喉咙干&#xff0c;感觉整个…

hcip什么时候考试?一文带您了解hcip考试报名与预约流程

其他考试一般都会有固定的时间&#xff0c;但hcip不一样&#xff0c;它的考试时间并不固定&#xff0c;这就让考生很是疑惑&#xff1a;hcip什么时候考试呢?除了知道考试时间之外&#xff0c;还要了解hcip的报名条件、报名流程等相关内容。关于这些问题的答案&#xff0c;小编…

blender图像如何分层导出?blender动画云渲染

在blender渲染时产品会被其他物体影响&#xff0c;这时候就可以用到blender中的阻隔&#xff1b;分层导出图像到PS中进行校色等后期处理。 在分层前&#xff0c;我们需要先打开渲染属性-胶片-透明&#xff0c;这样导出的图像才是透明背景的&#xff0c;反之会变成黑色底。 第一…