java获取服务器配置文件的值

获取服务器配置文件的值

    @Value("${bpm.minio.bucketName:}")
    private String minioBarrel;

在这里插入图片描述

获取nacos 配置文件的值

  • minio配置类
package com.guodi.bpm.common.minio;

import org.springframework.boot.context.properties.ConfigurationProperties;


/**
 * minio配置
 *
 * @author lwh
 * @since 2023-03-21
 */
@ConfigurationProperties(prefix = "bpm.minio")
public class MinioProperties {

    /**
     * MinIO地址
     */
    private String endpoint;

    /**
     * 访问的key
     */
    private String accessKey;

    /**
     * 访问的秘钥
     */
    private String secretKey;

    /**
     * 存储桶名称
     */
    private String bucketName;

    public String getEndpoint() {
        return endpoint;
    }

    public void setEndpoint(String endpoint) {
        this.endpoint = endpoint;
    }

    public String getAccessKey() {
        return accessKey;
    }

    public void setAccessKey(String accessKey) {
        this.accessKey = accessKey;
    }

    public String getSecretKey() {
        return secretKey;
    }

    public void setSecretKey(String secretKey) {
        this.secretKey = secretKey;
    }

    public String getBucketName() {
        return bucketName;
    }

    public void setBucketName(String bucketName) {
        this.bucketName = bucketName;
    }
}

  • minio实现类
package com.guodi.bpm.service.impl;

import com.guodi.bpm.common.auth.secure.utils.FileUtil;
import com.guodi.bpm.common.auth.secure.utils.Func;
import com.guodi.bpm.common.auth.secure.utils.StringPool;
import com.guodi.bpm.common.auth.secure.utils.StringUtil;
import com.guodi.bpm.common.minio.MinioException;
import com.guodi.bpm.common.minio.MinioProperties;
import com.guodi.bpm.common.minio.model.MinioFile;
import com.guodi.bpm.common.minio.model.StorageObject;
import com.guodi.bpm.service.IFileService;
import io.minio.*;
import io.minio.messages.Item;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.util.CollectionUtils;

import java.io.InputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.sql.Date;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Optional;

/**
 * minio文件获取存储实现类
 *
 * @author 梁伟浩
 * @date 2023/9/13 11:06
 * @study 星期三
 */
@EnableConfigurationProperties({MinioProperties.class})
public class MinIOFileServiceImpl implements IFileService {


    private static final Logger LOGGER = LoggerFactory.getLogger(MinIOFileServiceImpl.class);

    public static final String DEFAULT_ENCODED = "UTF-8";

    private final String bucketName;

    private final MinioClient minioClient;

    private final String endpoint;


    public MinIOFileServiceImpl(MinioProperties minioProperties) {
        this.minioClient = this.getMinioClient(minioProperties);
        this.bucketName = minioProperties.getBucketName();
        this.endpoint = minioProperties.getEndpoint();
    }


    public MinioClient getMinioClient(MinioProperties minioProperties) {
        String endpoint = minioProperties.getEndpoint();
        String accesskey = minioProperties.getAccessKey();
        String secretkey = minioProperties.getSecretKey();
        String bucketName = minioProperties.getBucketName();
        if (endpoint == null || "".equals(endpoint)) {
            throw new MinioException("Minio的URL未在application.yml配置!");
        }
        if (accesskey == null || "".equals(accesskey)) {
            throw new MinioException("Minio用户名未在application.yml配置!");
        }
        if (secretkey == null || "".equals(secretkey)) {
            throw new MinioException("Minio密码未在application.yml配置!");
        }
        if ((bucketName == null || "".equals(bucketName))) {
            throw new MinioException("存储桶名称未在application.yml配置!");
        }
        MinioClient minioClient =
                MinioClient.builder()
                        .endpoint(endpoint)
                        .credentials(accesskey, secretkey)
                        .build();

        makeBucket(minioClient, bucketName);
        return minioClient;
    }


    /**
     * 创建minio存储桶
     *
     * @param minioClient minio客户端
     * @param bucketName  桶名称
     */
    private void makeBucket(MinioClient minioClient, String bucketName) {
        try {
            //创建一个MinIO的Java客户端
            boolean isExist = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build());
            if (!isExist) {
                //创建存储桶
                minioClient.makeBucket(MakeBucketArgs.builder().bucket(bucketName).build());
                //设置桶策略
                String policyJson =
                        "{\n" +
                                "     \"Statement\": [\n" +
                                "         {\n" +
                                "             \"Action\": [\n" +
                                "                 \"s3:GetBucketLocation\",\n" +
                                "                 \"s3:ListBucket\"\n" +
                                "             ],\n" +
                                "             \"Effect\": \"Allow\",\n" +
                                "             \"Principal\": \"*\",\n" +
                                "             \"Resource\": \"arn:aws:s3:::" + bucketName + "\"\n" +
                                "         },\n" +
                                "         {\n" +
                                "             \"Action\": [\n" +
                                "                 \"s3:GetObject\",\n" +
                                "                 \"s3:PutObject\"\n" +
                                "             ],\n" +
                                "             \"Effect\": \"Allow\",\n" +
                                "             \"Principal\": \"*\",\n" +
                                "             \"Resource\": \"arn:aws:s3:::" + bucketName + "/*\"\n" +
                                "         }\n" +
                                "     ],\n" +
                                "     \"Version\": \"2012-10-17\"\n" +
                                " }";
                minioClient.setBucketPolicy(
                        SetBucketPolicyArgs.builder().bucket(bucketName).config(policyJson).build());
            }
        } catch (Exception e) {
            throw new MinioException("创建minio存储桶异常", e);
        }
    }

    /**
     * 添加 存储对象
     *
     * @param objectName  存储对象名称
     * @param inputStream 输入流
     * @return url
     */
    @Override
    public String putObject(String objectName, InputStream inputStream) {
        if ((objectName == null || "".equals(objectName))) {
            throw new MinioException("存储对象名称objectName不能为空!");
        }
        try {
            minioClient.putObject(
                    PutObjectArgs.builder().bucket(bucketName).object(objectName).stream(
                            inputStream, -1, 10485760)
                            .build());
            LOGGER.info("文件上传成功!");
            return URLDecoder.decode(this.getObjectUrl(objectName), "utf-8");
        } catch (Exception e) {
            throw new MinioException(e.getMessage(), e);
        }
    }

    /**
     * 获取 存储对象
     *
     * @param objectName 存储对象名称
     * @return inputStream
     */
    @Override
    public InputStream getObject(String objectName) {
        InputStream inputStream = null;
        try {
            inputStream = minioClient.getObject(
                    GetObjectArgs.builder()
                            .bucket(bucketName)
                            .object(objectName)
                            .build());
        } catch (Exception e) {
            throw new MinioException("minio获取储存对象路径不存在!", e.getCause());
        }
        return inputStream;
    }


    /**
     * @描述: 获取文件大小
     * @入参: fileUrl:文件访问全路径
     * @出参: 文件大小
     * @作者: 肖俊杰
     * @日期: 2021/11/4 15:36
     **/
    @Override
    public Long getFileSize(String fileUrl) {
        Long fileSize = 0L;
        try {
            if (fileUrl.indexOf(bucketName) != -1) {
                fileUrl = StringUtil.sub(fileUrl,
                        fileUrl.indexOf(bucketName) + bucketName.length(), -1);
            }
            MinioFile minioFile = this.getMinioFile(fileUrl);
            if (minioFile != null) {
                fileSize = minioFile.getLength();
            }
        } catch (Exception e) {
            e.printStackTrace();
            LOGGER.error("下载发生错误: {}!", e.getMessage());
            return fileSize;
        }
        return fileSize;
    }

    /**
     * 获取 存储对象
     *
     * @param objectName   存储对象名称
     * @param outputStream 输出流
     */
    @Override
    public void getObject(String objectName, OutputStream outputStream) {
        InputStream is = getObject(objectName);
        Optional.ofNullable(is).ifPresent(inputStream -> {
            try {
                byte[] buffer = new byte[2048];
                int len = 0;
                while ((len = inputStream.read(buffer)) != -1) {
                    outputStream.write(buffer, 0, len);
                }
                outputStream.flush();
            } catch (Exception e) {
                throw new MinioException(e.getMessage(), e);
            }
        });
    }

    /**
     * 获取对象列表
     *
     * @param objectPrefix 对象前缀
     * @return 对象列表
     * @author 陈国勇
     * @since 2023--4-18
     */
    @Override
    public List<StorageObject> listObjects(String objectPrefix) throws MinioException {
        List<StorageObject> storageObjects = new ArrayList<>();
        try {
            Iterable<Result<Item>> objects = minioClient.listObjects(
                    ListObjectsArgs.builder()
                            .bucket(bucketName)
                            .prefix(objectPrefix)
                            .recursive(true)
                            .build()
            );
            for (Result<Item> itemResult : objects) {
                Item item = itemResult.get();
                StorageObject storageObject = new StorageObject();
                storageObject.setBucketName(bucketName);
                storageObject.setObjectName(item.objectName());
                storageObjects.add(storageObject);
            }
        } catch (Exception e) {
            throw new MinioException(e.getMessage(), e);
        }
        return storageObjects;
    }

    /**
     * 根据存储对象名称 删除 存储对象
     *
     * @param objectName 存储对象名称
     */
    @Override
    public void removeObject(String objectName) {
        try {
            minioClient.removeObject(RemoveObjectArgs.builder().bucket(bucketName).object(objectName).build());
        } catch (Exception e) {
            throw new MinioException(e.getMessage(), e);
        }
    }

    /**
     * 根据文件访问全路径删除 存储对象
     *
     * @param fileUrl :文件访问全路径地址
     */
    @Override
    public void removeByFileUrl(String fileUrl) {
        try {
            String downloadUrl = fileUrl.substring(fileUrl.indexOf(bucketName) + bucketName.length());
            minioClient.removeObject(RemoveObjectArgs.builder().bucket(bucketName).object(downloadUrl).build());
        } catch (Exception e) {
            throw new MinioException(e.getMessage(), e);
        }
    }

    /**
     * 批量删除 存储对象
     *
     * @param objectNames 存储对象名称集合
     */
    @Override
    public void removeObjectBatch(Collection<String> objectNames) {
        if (!CollectionUtils.isEmpty(objectNames)) {
            objectNames.forEach(objectName -> {
                this.removeObject(objectName);
            });
        }
    }

    /**
     * 下载 存储对象
     *
     * @param objectName 存储对象名称
     * @param fileName   下载存储路径及文件名
     */
    @Override
    public void downloadObject(String objectName, String fileName) {
        try {
            minioClient.downloadObject(
                    DownloadObjectArgs.builder()
                            .bucket(bucketName)
                            .object(objectName)
                            .filename(fileName).build());
        } catch (Exception e) {
            throw new MinioException(e.getMessage(), e);
        }
    }

    /**
     * 根据存储对象名称获取Url
     *
     * @param objectName 存储对象名称
     * @return url
     */
    @Override
    public String getObjectUrl(String objectName) {
        try {
            return minioClient.getObjectUrl(bucketName, objectName);
        } catch (Exception e) {
            throw new MinioException(e.getMessage(), e);
        }
    }

    /**
     * 根据存储对象名称获取Mino文件对象
     *
     * @param objectName 存储对象名称
     * @return MinioFile
     */
    private MinioFile getMinioFile(String objectName) {
        MinioFile minoFile = new MinioFile();
        try {
            ObjectStat stat = minioClient.statObject(StatObjectArgs.builder().bucket(bucketName).object(objectName).build());
            String name = Func.isEmpty(stat.name()) ? StringUtil.sub(objectName, objectName.lastIndexOf("/") + 1
                    , objectName.length()) : StringUtil.sub(stat.name(), stat.name().lastIndexOf("/") + 1
                    , stat.name().length());
            minoFile.setName(name);
            minoFile.setLink(fileLink(stat.name()));
            minoFile.setHash(String.valueOf(stat.hashCode()));
            minoFile.setLength(stat.length());
            minoFile.setPutTime(Date.from(stat.createdTime().toInstant()));
            minoFile.setContentType(stat.contentType());
            minoFile.setSuffix(name.substring(name.lastIndexOf(".") + 1));
        } catch (Exception e) {
            LOGGER.error("读取存储对象异常", e);
            throw new MinioException("读取存储对象异常", e);
        }
        return minoFile;
    }

    @Override
    public String fileLink(String fileName) {
        return StringPool.SLASH.concat(bucketName).concat(StringPool.SLASH).concat(fileName);
    }

    /**
     * @param link
     * @描述 文件路径截取、转码以及格式化
     * <p>
     * 截取:全路径会截取为相对路径,示例:http://10.3.1.60:9000/guodi/1.png 变成 /guodi/1.png
     * 转码:中文乱码直接转成UTF-8编码
     * 格式化:文件路径中正斜杠会替换成反斜杠
     * </p>
     * @作者 肖俊杰
     * @日期 2022/1/6 9:54
     * @入参 link
     * @出参 String
     */
    @Override
    public String formatLink(String link) {
        if (StringUtil.isBlank(link)) {
            return null;
        }
        if (link.indexOf(endpoint) != -1) {
            link = StringUtil.sub(link, link.indexOf(endpoint) + endpoint
                    .length());
        }
        try {
            link = FileUtil.formatUrl(link);
            link = URLDecoder.decode(link, DEFAULT_ENCODED);
        } catch (UnsupportedEncodingException e) {
            LOGGER.error(e.getMessage());
        }
        return link;
    }
    
  /**
     * 获取储存桶名称
     * @作者 梁伟浩
     * @日期 2024/1/26 10:12
     * @study 星期五
     * @return bucketName
     */
    @Override
    public String getBucketName() {
        return bucketName;
    }
}

在这里插入图片描述

<think>嗯,用户让我详细介绍Redis的详细使用说明。首先,我需要确定用户的需求是什么。可能他们是一个开发者,刚开始接触Redis,或者想深入了解它的高级功能。接下来,我应该从基础开始,逐步深入,覆盖主要功能和常见使用场景。 首先,Redis是什么?它是一个内存数据结构存储,支持多种数据结构,比如字符串、哈希、列表等等。需要解释它的主要特点,比如高性能、持久化、支持事务和发布订阅等。然后,可能需要分几个部分来介绍:安装配置、数据结构、持久化、集群、应用场景等等。 安装部分,可能需要给出不同操作系统的安装步骤,比如Ubuntu用apt-get,CentOS用yum,macOS用Homebrew。然后基本命令,比如启动服务端和客户端,还有常用的配置项,比如端口、持久化设置。 数据结构是Redis的核心,每个结构都需要详细说明,比如字符串的常用命令set、get,哈希的hset、hget,列表的lpush、rpop,集合的交并差操作,有序集合的排序,还有位图和地理空间这些高级结构。需要给出每个数据结构的典型使用场景,比如缓存、计数器、消息队列等。 持久化部分,用户可能会关心RDB和AOF的区别,以及如何配置。需要解释两者的优缺点,比如RDB适合备份,AOF更安全但可能更慢。事务和Lua脚本也是重要部分,说明事务的执行方式和注意事项,比如没有回滚,以及Lua脚本的原子性。 集群和高可用方面,主从复制、哨兵、Cluster模式需要介绍,特别是Cluster的自动分片和故障转移。性能优化部分,内存管理和批量操作是关键,比如使用Pipeline减少网络延迟,避免大键等。 应用场景要具体,比如缓存、会话存储、排行榜、消息队列,这些例子帮助用户理解如何在实际项目中使用Redis。安全性方面,密码认证和网络隔离也是必须提到的。 最后,注意事项部分,内存不足的问题,持久化数据安全,以及集群配置的复杂性,都是用户可能遇到的挑战。需要提醒他们根据需求选择合适的策略。 可能用户没有明说的需求是,他们可能在设计系统时需要选择合适的技术栈,或者遇到了性能问题想优化。所以除了基本介绍,还需要强调最佳实践和常见陷阱,比如避免大键、合理选择数据结构。另外,安全性容易被忽视,特别是生产环中设置密码的重要性。 总之,结构需要清晰,由浅入深,覆盖基础到高级内容,同时结合实际例子,帮助用户更好地理解和应用Redis。可能还需要检查是否有最新的Redis版本更新内容,确保信息准确。比如最新版本是否引入了新数据结构或命令,但目前考虑用户可能使用的是较常见版本,所以先覆盖主流功能。以下</think>是对 **Redis** 的详细介绍和使用说明,内容涵盖核心概念、使用场景、数据结构及操作指南等: --- ### **1. Redis 简介** Redis(**Re**mote **Di**ctionary **S**ervice)是一个开源的 **内存数据结构存储系统**,支持多种数据结构(如字符串、哈希、列表等),常用于缓存、消息队列、实时数据处理等场景。其特点包括: - **高性能**:数据存储在内存中,读写速度极快(微秒级响应)。 - **持久化**:支持 RDB(快照)和 AOF(追加日志)两种持久化方式。 - **高可用**:通过哨兵(Sentinel)和集群(Cluster)模式实现故障转移和扩展。 - **多数据结构**:支持字符串、列表、哈希、集合、有序集合等。 --- ### **2. 安装启动** #### **安装 Redis** - **Linux (Ubuntu/Debian)**: ```bash sudo apt-get install redis-server ``` - **macOS**: ```bash brew install redis ``` - **Windows**:官方不支持原生版本,可通过 WSL 或 Docker 运行。 #### **启动 Redis** - **启动服务端**: ```bash redis-server ``` - **启动客户端**: ```bash redis-cli ``` #### **基本配置** 配置文件通常位于 `/etc/redis/redis.conf`,常用配置项: - `port 6379`:默认端口号。 - `bind 127.0.0.1`:绑定 IP。 - `save 60 1000`:60 秒内有 1000 次修改则触发 RDB 持久化。 - `appendonly yes`:启用 AOF 持久化。 --- ### **3. 核心数据结构操作** #### **(1) 字符串(String)** - **用途**:缓存、计数器、分布式锁。 - **常用命令**: ```bash SET key value # 设置值 GET key # 获取值 INCR key # 自增计数器 EXPIRE key 60 # 设置过期时间(秒) ``` #### **(2) 哈希(Hash)** - **用途**:存储对象(如用户信息)。 - **常用命令**: ```bash HSET user:1 name "Alice" age 30 # 设置字段 HGET user:1 name # 获取字段值 HGETALL user:1 # 获取所有字段 ``` #### **(3) 列表(List)** - **用途**:消息队列、最新动态。 - **常用命令**: ```bash LPUSH queue task1 # 左侧插入元素 RPOP queue # 右侧弹出元素 LRANGE queue 0 -1 # 获取全部元素 ``` #### **(4) 集合(Set)** - **用途**:标签、好友关系、去重。 - **常用命令**: ```bash SADD tags "redis" "db" # 添加元素 SMEMBERS tags # 获取所有元素 SINTER set1 set2 # 求交集 ``` #### **(5) 有序集合(Sorted Set)** - **用途**:排行榜、优先级队列。 - **常用命令**: ```bash ZADD rank 100 "user1" # 添加带分数的元素 ZRANGE rank 0 -1 WITHSCORES # 按分数升序获取 ZREVRANGE rank 0 10 # 按分数降序获取前10 ``` #### **(6) 其他数据结构** - **位图(Bitmap)**:用于二值统计(如每日签到)。 - **HyperLogLog**:基数统计(如 UV 去重)。 - **地理空间(GEO)**:存储经纬度,支持距离计算。 --- ### **4. 持久化高可用** #### **(1) 持久化方式** - **RDB**:定期生成数据快照,恢复速度快,但可能丢失部分数据。 - **AOF**:记录所有写操作命令,数据更安全,但文件较大。 #### **(2) 主从复制** 配置从节点同步主节点数据: ```bash # 在从节点配置文件中添加: replicaof <master-ip> <master-port> ``` #### **(3) 哨兵模式(Sentinel)** 实现自动故障转移: ```bash # 哨兵配置文件 sentinel.conf sentinel monitor mymaster 127.0.0.1 6379 2 sentinel down-after-milliseconds mymaster 5000 ``` #### **(4) 集群模式(Cluster)** 分片存储数据,支持横向扩展: ```bash # 创建集群(至少3主3从) redis-cli --cluster create 127.0.0.1:7000 127.0.0.1:7001 ... ``` --- ### **5. 典型应用场景** - **缓存加速**:缓存数据库查询结果,降低后端负载。 - **会话存储(Session Store)**:集中管理用户会话。 - **排行榜**:使用有序集合实时更新排名。 - **消息队列**:通过列表或 Stream 数据结构实现。 - **分布式锁**:通过 `SET key value NX EX` 实现互斥锁。 --- ### **6. 性能优化注意事项** - **内存管理**:监控内存使用,避免 OOM(`maxmemory` 配置)。 - **Pipeline 批量操作**:减少网络往返次数。 - **避免大 Key**:单个 Key 的 Value 不宜过大(如超过 10KB)。 - **合理选择数据结构**:例如用 Hash 代替多个 String 存储对象。 --- ### **7. 安全性** - **密码认证**:在配置文件中设置 `requirepass yourpassword`。 - **禁用危险命令**:禁止使用 `FLUSHALL`、`CONFIG` 等。 - **网络隔离**:通过防火墙限制访问 IP。 --- ### **8. 常见问题** - **缓存穿透**:使用布隆过滤器或缓存空值。 - **缓存雪崩**:设置随机过期时间,保证高可用架构。 - **数据一致性**:结合数据库的 Binlog 或延迟双删策略。 --- 通过以上内容,可以全面掌握 Redis 的核心功能和使用技巧。实际应用中需根据场景选择合适的数据结构和架构模式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Java中的战斗机

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值