2024-08-27



public class RDelayedQueueUsageExample {
    public static void main(String[] args) throws InterruptedException {
        // 创建Redisson实例
        RedissonClient redisson = Redisson.create();
 
        // 获取延迟队列RDelayedQueue对象
        RDelayedQueue<String> delayedQueue = redisson.getDelayedQueue();
 
        // 创建一个延迟任务(例如5秒后执行)
        delayedQueue.offer("message-1", 5, TimeUnit.SECONDS);
 
        // 创建一个延迟任务并指定执行时间点(Date类型)
        Date futureDate = new Date(System.currentTimeMillis() + 5000);
        delayedQueue.offer("message-2", futureDate);
 
        // 创建一个延迟消费者(监听器)
        delayedQueue.addListener(new MessageListener<String>() {
            @Override
            public void onMessage(String message, long delay) {
                // 处理消息
                System.out.println("Process delayed message: " + message);
            }
        });
 
        // 延迟消费者将会在消息到达时自动处理,此处主线程休眠以等待消息处理
        Thread.sleep(10000);
 
        // 关闭Redisson客户端
        redisson.shutdown();
    }
}

这段代码展示了如何使用Redisson提供的RDelayedQueue来处理延迟消息。它创建了一个Redisson客户端,获取了一个RDelayedQueue对象,并向队列中添加了两个延迟任务。然后,它添加了一个监听器来处理这些任务,并在主线程中等待一段时间以便可以看到消息的处理结果。最后,它关闭了Redisson客户端。这个例子简单直观地展示了如何使用Redisson的RDelayedQueue来进行延迟消息处理。

2024-08-27



# Redis 配置文件示例
 
# 设置Redis监听的IP地址,如果要监听所有地址,可以使用0.0.0.0
bind 127.0.0.1
 
# 设置Redis服务的监听端口,默认为6379
port 6379
 
# 设置Redis是否以守护进程方式运行
daemonize no
 
# 设置Redis的日志文件路径
logfile "/var/log/redis/redis-server.log"
 
# 设置Redis数据库的数量,默认16个数据库(db0...db15)
databases 16
 
# 设置Redis密码,如果配置了密码,客户端连接时需要使用AUTH命令进行认证
requirepass yourpassword
 
# 设置Redis最大内存容量,达到最大内存后,会使用allkeys-lru或volatile-lru策略进行数据淘汰
maxmemory 2gb
 
# 设置内存淘汰策略,当达到maxmemory时使用的策略
maxmemory-policy allkeys-lru
 
# 设置是否开启虚拟内存机制,需要vm-enabled yes配合使用
vm-enabled no
 
# 设置swap文件路径
dir /var/spool/redis
 
# 设置是否在每次后台I/O操作后进行日志记录,如RDB/AOF
appendonly no
 
# 设置AOF文件的更新频率
appendfsync everysec
 
# 设置客户端空闲超时时间,0表示禁用空闲连接超时
timeout 300
 
# 设置是否启用TCP keepalive,可以防止死链接
tcp-keepalive 0
 
# 设置Redis连接数的最大客户端数,返回错误当达到这个数值
maxclients 10000
 
# 设置Redis集群配置参数,如果不使用集群可以忽略
cluster-enabled no
cluster-config-file nodes.conf
cluster-node-timeout 5000
 
# 设置Redis的高级配置参数,如RDB保存点,大Key检测等
save 900 1
save 300 10
save 60 10000
 
# 设置检测大键的阈值,超过这个值会报告为大Key
large-key-threshold 10
 
# 设置是否记录慢查询,并可以配置慢查询的阈值
slowlog-log-slower-than 10000
slowlog-max-len 128
 
# 设置Redis的PID文件路径
pidfile /var/run/redis/redis-server.pid
 
# 设置Redis的主从复制配置
slaveof <masterip> <masterport>

这个配置文件提供了基本的Redis配置选项,包括监听地址、端口、日志文件、数据库数量、密码、内存大小、虚拟内存、磁盘持久化策略、连接超时、集群配置、保存点和慢查询日志配置等。开发者可以根据自己的需求进行相应的调整。

2024-08-27

Redis 提供了两种持久化方式:RDB(Redis DataBase)和AOF(Append Only File)。

  1. RDB 持久化:

    RDB 是 Redis 默认的持久化方式。它会在一定的间隔时间内将内存中的数据集快照写入磁盘,生成一个dump.rdb文件。

    配置文件中的关键配置项:

    
    
    
    save 900 1        # 900秒内至少1个键被修改则触发保存
    save 300 10       # 300秒内至少10个键被修改则触发保存
    save 60 10000     # 60秒内至少10000个键被修改则触发保存
    dbfilename dump.rdb          # RDB文件名
    dir ./                        # RDB文件存储目录
  2. AOF 持久化:

    AOF 持久化是通过保存 Redis 服务器所执行的写命令来记录数据库状态的。

    配置文件中的关键配置项:

    
    
    
    appendonly yes          # 开启AOF持久化存储
    appendfilename "appendonly.aof"  # AOF文件名
    appendfsync everysec    # 每秒同步一次至磁盘

在实际应用中,可以根据数据的重要性和性能需求选择合适的持久化方式,或者两者结合使用。如果需要快速恢复数据,可以使用 RDB 文件。如果需要保证数据的完整性,可以使用 AOF 方式。

2024-08-27

AOF(Append Only File)是Redis的另一种持久化方式,用于记录每一个写操作,将Redis执行过的所有写指令记录下来(读操作不记录),只允许追加不允许改写,重启时重新执行AOF文件中的命令达到数据恢复的目的。

AOF文件大小会一直增大,为了解决这个问题,Redis引入了AOF文件重写机制。当AOF文件的大小超过了配置的阈值时,Redis会自动启动AOF文件的内容压缩,只保留可以恢复数据的最小指令集。

AOF文件重写的主要目的是为了减少AOF文件的大小,提高磁盘效率。

以下是AOF文件重写的伪代码:




def rewriteAof():
    # 创建临时文件
    temp_file = create_temp_file()
    
    # 遍历现有的所有键
    for key in db.keys():
        # 对每个键取值
        value = db.get(key)
        # 将键值对写入临时文件
        temp_file.write(key, value)
    
    # 替换原有的AOF文件
    replace_aof_file(temp_file)

在Redis中,AOF文件重写是通过bgrewriteaof命令触发的。当AOF文件的大小超过了配置的阈值时,Redis会在后台自动执行AOF文件的重写操作。如果手动执行该命令,Redis会创建一个新的AOF文件,只包含当前数据库状态所必需的最小指令集。这样可以显著减小AOF文件的大小,加快数据恢复的速度。

2024-08-27



import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Service;
 
@Service
public class RedisService {
 
    @Autowired
    private StringRedisTemplate stringRedisTemplate;
 
    public void setKey(String key, String value) {
        stringRedisTemplate.opsForValue().set(key, value);
    }
 
    public String getKey(String key) {
        return stringRedisTemplate.opsForValue().get(key);
    }
 
    public void deleteKey(String key) {
        stringRedisTemplate.delete(key);
    }
}

这段代码展示了如何在Spring Boot 3项目中使用StringRedisTemplate来进行基本的Redis操作,包括设置键值对、获取键对应的值以及删除键。这是一个简单的服务类,可以被其他组件注入并使用。

2024-08-27

清空 Redis 的 LIST 有多种实现方法,以下是几种不同的方式:

方法一:使用 DEL 命令删除 LIST




redis_conn = redis.Redis(host='localhost', port=6379)
redis_conn.delete('my_list')

方法二:使用 LTRIM 命令重新赋值一个空的 LIST




redis_conn = redis.Redis(host='localhost', port=6379)
redis_conn.ltrim('my_list', 1, 0)

方法三:使用 RPUSHX 命令将空列表写入 LIST




redis_conn = redis.Redis(host='localhost', port=6379)
redis_conn.rpushx('my_list', [])

方法四:使用 LPUSH、RPOP 或 LTRIM 结合循环删除 LIST 中的元素




redis_conn = redis.Redis(host='localhost', port=6379)
while redis_conn.llen('my_list') > 0:
    redis_conn.rpop('my_list')

以上是几种不同的方法来清空 Redis 的 LIST,可以根据具体情况选择适应的方式。

2024-08-27

以下是一个简化的RedisUtils类示例,展示了如何在Spring Boot 3项目中引入Redis并封装常用操作。




import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;
 
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
 
@Component
public class RedisUtils {
 
    @Autowired
    private RedisTemplate<String, Object> redisTemplate;
 
    // 存储数据
    public void set(String key, Object value) {
        redisTemplate.opsForValue().set(key, value);
    }
 
    // 存储数据并设置过期时间
    public void set(String key, Object value, long timeout, TimeUnit unit) {
        redisTemplate.opsForValue().set(key, value, timeout, unit);
    }
 
    // 获取数据
    public Object get(String key) {
        return redisTemplate.opsForValue().get(key);
    }
 
    // 删除数据
    public void delete(String key) {
        redisTemplate.delete(key);
    }
 
    // 批量删除数据
    public void delete(Collection<String> keys) {
        redisTemplate.delete(keys);
    }
 
    // 批量获取数据
    public List<Object> multiGet(Collection<String> keys) {
        return redisTemplate.opsForValue().multiGet(keys);
    }
 
    // 判断数据是否存在
    public boolean hasKey(String key) {
        return redisTemplate.hasKey(key);
    }
 
    // 获取过期时间
    public Long getExpire(String key) {
        return redisTemplate.getExpire(key, TimeUnit.SECONDS);
    }
 
    // 设置过期时间
    public void setExpire(String key, long timeout, TimeUnit unit) {
        redisTemplate.expire(key, timeout, unit);
    }
 
    // 自增操作
    public Long increment(String key, long delta) {
        return redisTemplate.opsForValue().increment(key, delta);
    }
 
    // 自减操作
    public Long decrement(String key, long delta) {
        return redisTemplate.opsForValue().increment(key, -delta);
    }
 
    // Hash操作
    public void hSet(String key, String hashKey, Object value) {
        redisTemplate.opsForHash().put(key, hashKey, value);
    }
 
    public Object hGet(String key, String hashKey) {
        return redisTemplate.opsForHash().get(key, hashKey);
    }
 
    public void hDelete(String ke
2024-08-27

Redis的配置文件是一个文本文件,通常名为redis.conf。它包含了Redis服务器的各种配置指令,比如设置Redis监听的端口、设置数据库的数量、设置持久化策略等。

以下是一个基本的redis.conf配置文件的例子:




# Redis配置文件示例
 
# 设置Redis监听的端口,默认为6379
port 6379
 
# 设置Redis监听的网络接口,如果需要让Redis服务器监听所有接口,请使用0.0.0.0
# bind 127.0.0.1
 
# 设置Redis是否以守护进程方式运行
daemonize no
 
# 设置Redis的日志文件路径
logfile "/var/log/redis/redis-server.log"
 
# 设置数据库数量,默认16个数据库 (0...15)
databases 16
 
# 设置密码认证,要启用此功能,需要将此行改为 "requirepass yourpassword"
# requirepass foobared
 
# 设置持久化策略
save 900 1
save 300 10
save 60 10000
 
# 设置快照文件名
dbfilename dump.rdb
 
# 设置快照文件和AOF文件的目录
dir ./
 
# 设置Redis最大内存使用量,达到上限时会触发键的淘汰策略
maxmemory <bytes>
 
# 设置Redis的主从复制配置
# slaveof <masterip> <masterport>
 
# 设置Redis连接密码,主服务器设置该项后,从服务器连接主服务器时需要使用该密码
# masterauth <master-password>
 
# 其他配置项...

要修改Redis的配置,你可以直接编辑redis.conf文件,然后重启Redis服务。例如,要改变监听端口,你可以找到port 6379这一行,然后将6379改为你想要的端口号。

注意:在生产环境中,应该通过Redis的命令行接口或者使用CONFIG SET命令来设置配置项,而不是直接编辑配置文件,因为这样可以在不中断服务的情况下动态更新配置。

2024-08-27



import redis
 
# 连接到Redis服务器
redis_client = redis.StrictRedis(host='localhost', port=6379, db=0)
 
# 订阅频道
channel = "chat"
pubsub = redis_client.pubsub()
pubsub.subscribe(channel)
 
print(f"订阅频道 {channel} 成功,等待消息...")
 
try:
    for message in pubsub.listen():
        if message['type'] == 'message':
            print(f"接收到消息: {message['data'].decode()}")
except KeyboardInterrupt:
    pubsub.unsubscribe(channel)
    print(f"取消订阅频道 {channel}")

这段代码演示了如何使用redis-py库来订阅一个Redis频道,并在收到消息时进行处理。当运行这段代码时,它会连接到Redis服务器,订阅指定的频道,并打印接收到的消息。按下Ctrl+C可以取消订阅并退出程序。

2024-08-27

使用Redis的SETNX命令实现分布式锁是一个常见的方案。SETNX是"SET if Not eXists"的缩写,也就是只有当键不存在时才设置值。这个特性使得它可以被用作锁的机制。

以下是一个使用Python和redis-py库实现的示例:




import redis
import time
import uuid
 
def acquire_lock(conn, lock_name):
    identifier = str(uuid.uuid4())  # 生成一个唯一的ID
    end = time.time() + 10  # 设置超时时间
 
    while time.time() < end:
        if conn.setnx(lock_name, identifier):  # 尝试获取锁
            return identifier
        time.sleep(0.001)
 
    return False
 
def release_lock(conn, lock_name, identifier):
    pipe = conn.pipeline(True)
    while True:
        try:
            pipe.watch(lock_name)
            if pipe.get(lock_name) == identifier:
                pipe.multi()
                pipe.delete(lock_name)
                pipe.execute()
                return True
            pipe.unwatch()
            break
        except redis.exceptions.WatchError:
            pass
    return False
 
# 使用示例
redis_conn = redis.StrictRedis(host='localhost', port=6379, db=0)
lock_name = "my_lock"
identifier = acquire_lock(redis_conn, lock_name)
if identifier:
    try:
        # 在这里执行需要互斥访问的代码
        print("Lock acquired")
    finally:
        # 确保释放锁
        if release_lock(redis_conn, lock_name, identifier):
            print("Lock released")
else:
    print("Could not acquire lock")

在这个示例中,acquire_lock 函数尝试获取锁,如果在10秒内未能获得锁,则返回Falserelease_lock 函数尝试释放锁,如果锁的标识符匹配,则删除键释放锁。这里使用了UUID生成唯一的锁标识符,避免了由于锁释放出现问题。