利用redis实现缓存、发布订阅、分布式锁功能

Redis是一个内存键值存储数据库,通常用于缓存、会话管理、消息队列等场景。以下是一些常见的Redis使用场景:

1.缓存:将常用的数据缓存在Redis中,以减少对数据库的访问次数,提高应用程序的性能。

2.会话管理:使用Redis来存储用户的会话数据,以提高应用程序的并发处理能力。

3.发布/订阅系统:使用Redis的发布/订阅功能来实现实时通知、消息推送等功能。

4.分布式锁:使用Redis的分布式锁来实现分布式系统中的互斥访问控制。

5.任务队列:使用Redis的列表或队列来实现异步任务处理、延迟任务等功能。

Jedis是Java开发人员常用的Redis客户端库,它提供了简单易用的API来操作Redis数据库,Jedis支持Redis的所有功能,包括字符串、列表、集合、有序集合等数据结构,以及事务、Lua脚本等高级功能。单同时需要注意Jedis采用了单线程模型,不能充分利用多核CPU的性能,Jedis需要手动管理连接池和资源回收等问题,容易出现内存泄漏等问题,Jedis缺乏异步API的支持,不能很好地处理高并发请求,Jedis的同步调用模型容易出现阻塞问题,需要使用异步模型或者连接池等技术来解决。Jedis本质上是直接连接的redis server,如果在多线程环境下是非线程安全的,这个时候只有使用连接池,为每个Jedis实例增加物理连接。

除了Jedis,Lettuce是一个高性能的Redis客户端库,基于Netty框架,提供了异步和响应式API来操作Redis数据库,且支持单机、主从、集群等多种Redis部署方式,另外,采用了连接池和自动重连等技术,能够自动管理连接和资源回收等问题。

前面是一些概念介绍,接下来看看,在实际项目中,如何采用spring boot和redis实现缓存机制。对于Spring boot,如果程序中没有定义类型为CacheManager的Bean组件或是名为cacheResolver的CacheResolver缓存解析器,Spring Boot将按顺序选择并启用以下缓存组件1.Generic,2.JCache (JSR-107)(EhCache 3、Hazelcast、Infinispan等),3.EhCache 2.x,4.Hazelcast,5.Infinispan,6.Couchbase,7.Redis,8.Caffeine,9.Simple。实际上,Spring Boot默认缓存管理中,没有添加任何缓存管理组件能实现缓存管理。因为开启缓存管理后,Spring Boot会按照上述列表顺序查找有效的缓存组件进行缓存管理,如果没有任何缓存组件,会默认使用最后一个Simple缓存组件进行管理。Simple缓存组件是Spring Boot默认的缓存管理组件,它默认使用内存中的ConcurrentMap进行缓存存储,所以在没有添加任何第三方缓存组件的情况下,可以实现内存中的缓存管理,但是不推荐使用这种缓存管理方式。当在Spring Boot默认缓存管理的基础上引入Redis缓存组件,即在pom.xml文件中添加Spring Data Redis依赖启动器后,SpringBoot会使用RedisCacheConfigratioin当做生效的自动配置类进行缓存相关的自动装配,容器中使用的缓存管理器是RedisCacheManager, 这个缓存管理器创建的Cache为RedisCache, 进而操控redis进行数据的缓存。所以,如果要采用redis实现缓存,首先需要引入“spring-boot-starter-data-redis”的依赖,接着在application.properties中配置redis服务相关的配置,如下图所示,另外,还需要创建RedisConfig配置类,配置类中定义cacheManager和redisCacheTemplate,如下图所示。

为什么需要在配置类中定义redisCacheTemplate和cacheManager呢?因为默认情况下,缓存的对象的类要实现Serializable接口,是以JDK序列化数据存储在Redis中,如果想实现JSON格式存入缓存中,那么就需要进行序列化,这里使用GenericJackson2JsonRedisSerializer类对value就行序列化。

最后在需要进行缓存的类或者方法上添加Cache相关的注解,即可实现缓存功能。常用注解以及使用场景如下所示:

@Cacheable:可以标记在一个方法上,也可以标记在一个类上。当标记在一个方法上时表示该方法是支持缓存的,当标记在一个类上时则表示该类所有的方法都是支持缓存的。对于一个支持缓存的方法,Spring会在其被调用后将其返回值缓存起来,以保证下次利用同样的参数来执行该方法时可以直接从缓存中获取结果,而不需要再次执行该方法。该注解一般用在查询方法上。

@CachePut:也可以声明一个方法支持缓存功能。与@Cacheable不同的是使用@CachePut标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,而是每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中。该注解一般用在新增方法上。

@CacheEvict:是用来标注在需要清除缓存元素的方法或类上的。当标记在一个类上时表示其中所有的方法的执行都会触发缓存的清除操作。该注解一般用在更新和删除方法上。

@EnableCaching:开启缓存功能,一般放在启动类上或者自定义的RedisConfig配置类上。部分代码片段如下所示,所有代码细节可以查看Demo。

下载Demo后,配置成本地Mysql的用户名、密码等信息,调用服务的post接口添加数据,再调用get接口获取数据,当从数据库中获取时,会打印的sql日志,具体如下所示。继续用get接口获取数据,则会从缓存中获取,日志中不再打印sql信息,当超过缓存失效时间20second后,又会从数据库中获取。

上面介绍了如何使用redis实现缓存,接下来看看发布/订阅功能,采用Redis实现发布订阅有很多优点,例如:

  1. 轻量级:Redis是一个内存数据库,所以它非常适合作为轻量级消息代理使用。因为Redis不需要像传统的消息中间件那样持久化数据,所以它可以非常快速地处理消息。

  1. 简单易用:使用Redis实现发布订阅非常简单,只需要几行代码就可以实现。

  1. 可扩展性:由于Redis可以作为集群部署,所以它的扩展性非常好。它可以轻松地处理大量消息,并且可以在需要时添加更多的Redis节点。

  1. 高性能:Redis是一个非常快速的内存数据库,所以它可以处理大量消息并提供快速的响应时间。

当然,相比于专门的消息中间件来传递message,消息中间件会有下面的一些优点。

  1. 持久性:与Redis不同,大多数消息中间件都提供消息持久化功能,这意味着即使应用程序在消息被发送后崩溃,消息也不会丢失。

  1. 多样性:消息中间件通常提供多种协议,例如AMQP、MQTT和STOMP等,可以让应用程序使用不同的协议进行通信。

  1. 可靠性:与Redis不同,大多数消息中间件具有复杂的可靠性机制,例如事务处理和消息确认,以确保消息被传递并正确处理。

  1. 高可用性:消息中间件通常提供集群部署,可以在节点故障时提供高可用性。

综上所述,使用Redis实现发布订阅是一个简单、轻量级、快速的解决方案,适合处理大量非关键性消息。而使用消息中间件则适用于更复杂的应用场景,例如需要可靠性保证、持久性和多样性的场景。那么如何使用spring boot,redis实现发布订阅功能呢?非常简单,具体步骤如下所示:

一:添加“spring-boot-starter-data-redis”的依赖

二:在application.properties文件中配置redis的host/port等信息

三:添加配置类信息和创建publish和subscribe服务,具体代码如下所示:在配置类中创建MessageListenerAdapter,MessageListenerAdapter是Spring AMQP中的一个类,它用于将消息侦听器(MessageListener)适配到处理具体消息的方法上。在代码的demo代码中是适配到onReceive()方法上。为什么需要配置MessageListenerAdapter呢?因为在Spring AMQP中,消息侦听器负责处理接收到的消息,但是要正确处理消息,必须了解消息的内容和格式,这使得编写消息处理逻辑变得复杂。MessageListenerAdapter就是为了简化这个过程而设计的。使用MessageListenerAdapter,可以将一个POJO对象的方法适配成消息侦听器,这样就不需要显式地编写MessageListener了。在适配过程中,MessageListenerAdapter将负责将消息转换为方法的参数,并将方法的返回值转换为消息。

此外,还配置了RedisMessageContainer,RedisMessageListenerContainer在Spring应用中使用Redis消息监听器时是必须配置的,它可以帮助自动管理Redis连接和连接池、自动订阅和取消订阅Redis频道或模式、自动分发Redis消息给相应的监听器处理,简化了代码的编写和维护,并确保了应用的可靠性和稳定性。

另外,还配置了ReactiveRedisTemplate,redistemplate是Spring Data Redis中提供的Redis客户端,用于进行与Redis的交互。

更多代码细节可查看demo,启动应用程序后,调用post接口"http://localhost:8080/api/news/publish",模拟往频道上发送消息,查看日志信息,可以看到订阅端打印了发送的消息,说明发布和订阅消息成功。

除了实现发布订阅功能,还可以借助redis实现分布式锁功能。分布式锁是在分布式系统中协调并发访问共享资源的一种常用机制,其主要目的是保证多个进程或线程在访问共享资源时的正确性和一致性。在分布式系统中,多个进程或线程可能会同时访问同一个共享资源,如数据库、缓存、文件等,如果不采取特殊的处理机制,可能会出现以下问题:

  1. 竞态条件:当多个进程或线程同时读写同一共享资源时,可能会出现互相干扰、顺序不确定等问题,导致程序运行不稳定或出现异常。

  1. 脏数据:当一个进程或线程正在修改某个共享资源时,如果另一个进程或线程也同时对其进行操作,可能会导致数据出现不一致的情况。

  1. 死锁:当多个进程或线程相互等待对方释放锁时,可能会出现死锁的情况,导致程序无法正常运行。

分布式锁就是解决上面这些问题的有效方法,实现的分布式锁应该具备如下的特点:

1、在分布式系统环境下,一个方法在同一时间只能被一个机器的一个线程执行;

2、高可用的获取锁与释放锁;

3、高性能的获取锁与释放锁;

4、具备可重入特性;

5、具备锁失效机制,防止死锁;

  1. 具备非阻塞锁特性,即没有获取到锁将直接返回获取锁失败。

常见的分布式锁实现方式包括基于数据库、ZooKeeper、Redis等技术实现,其中Redis是最常用的分布式锁实现方式之一。下面提供了利用redis实现分布式锁的部分代码片段。首先,创建RedisDistributedLock对象,在RedisDistributedLock中定义了tryLock()和unLock()方法,即添加锁和释放锁两个方法。对于tryLock(),实际是使用"redisTemplate.opsForValue().setIfAbsent"来实现添加锁的功能,该方法是 Spring Data Redis 提供的一个 Redis 命令封装方法,用于将 key/value 存储到 Redis 中,仅在该 key 不存在时才执行存储操作,并且该方法是线程安全的,可以用来实现分布式锁。释放锁是通过lua脚本实现,具体代码如下所示:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.data.redis.core.script.RedisScript;
import org.springframework.stereotype.Component;

import java.util.Collections;
import java.util.concurrent.TimeUnit;

@Component
public class RedisDistributedLock {

    private static final long DEFAULT_EXPIRE_TIME = 30000L; // 默认过期时间30秒
    private static final long DEFAULT_TRY_LOCK_TIMEOUT = 5000L; // 默认尝试获取锁的超时时间5秒
    private static final String UNLOCK_LUA_SCRIPT = "if redis.call('get', KEYS[1]) == ARGV[1] then\n" +
            "    return redis.call('del', KEYS[1])\n" +
            "else\n" +
            "    return 0\n" +
            "end";

    @Autowired
    private RedisTemplate<String, String> redisTemplate;

    /**
     * 尝试获取锁
     *
     * @param lockKey    锁的key
     * @param requestId  请求id,用于标识加锁的客户端
     * @param expireTime 锁的过期时间,单位毫秒
     * @param tryTimeout 尝试获取锁的超时时间,单位毫秒
     * @return 是否获取到锁
     */
    public boolean tryLock(String lockKey, String requestId, long expireTime, long tryTimeout) {
        long startTime = System.currentTimeMillis();
        while (System.currentTimeMillis() - startTime < tryTimeout) {
            Boolean locked = redisTemplate.opsForValue().setIfAbsent(lockKey, requestId, expireTime, TimeUnit.MILLISECONDS);
            if (locked != null && locked) {
                return true;
            }
            try {
                Thread.sleep(50);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        }
        return false;
    }

    /**
     * 尝试获取锁,使用默认的过期时间和尝试获取锁的超时时间
     *
     * @param lockKey   锁的key
     * @param requestId 请求id,用于标识加锁的客户端
     * @return 是否获取到锁
     */
    public boolean tryLock(String lockKey, String requestId) {
        return tryLock(lockKey, requestId, DEFAULT_EXPIRE_TIME, DEFAULT_TRY_LOCK_TIMEOUT);
    }

    /**
     * 释放锁
     *
     * @param lockKey   锁的key
     * @param requestId 请求id,用于标识加锁的客户端
     * @return 是否释放成功
     */
    public boolean unlock(String lockKey, String requestId) {
        RedisScript<Long> script = new DefaultRedisScript<>(UNLOCK_LUA_SCRIPT, Long.class);
        Long result = redisTemplate.execute(script, Collections.singletonList(lockKey), requestId);
        return result != null && result == 1;
    }
}

创建了RedisDistributedLock对象后,在实际业务代码中就可以调用添加锁和释放锁的方法完成分布式锁业务场景。在执行数据修改操作前调用tryLock()方法添加锁,执行完操作后,再释放锁。

@Autowired
private RedisDistributedLock redisDistributedLock;

public void doSomething(String key) {
    String token = redisDistributedLock.tryLock(key); // 尝试获取锁
    if (token == null) {
        // 获取锁失败
        return;
    }
    try {
        // 执行业务逻辑
    } finally {
        redisDistributedLock.releaseLock(key, token); // 释放锁
    }
}

以上就是通过redis实现分布式锁的主要代码片段,代码补充完整后,可通过性能测试工具并发访问接口,查看数据是否正常。

可以看到,采用spring boot框架,引入“spring-boot-starter-data-redis”依赖,借助已经封装好的注解或者对象,可以很方便的通过redis实现缓存、发布订阅、分布式锁功能。

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
Redis从入门到高可用 分布式实战教程,共140多节课程、 掌握redis主从、哨兵、集群 ,参数调优 目录: 9-9 原生安装-1.准备节点.mp4 9-8 原生安装.mp4 9-7 基本架构.mp4 9-6 虚拟槽哈希分布.mp4 9-5 一致性哈希分区.mp4 9-4 节点取余分区.mp4 9-3 数据分布概论.mp4 9-2 呼唤集群.mp4 9-16 原生命令和redis-trib.rb对比.mp4 9-14 ruby环境准备-操作.mp4 9-13 ruby环境准备-说明.mp4 9-12 原生安装-4.分配主从.mp4 9-10 原生安装-2.节点握手.mp4 9-1 本章目录.mp4 8-9 实现原理-1-故障转移演练.mp4 8-8 python客户端.mp4 8-7 java客户端.mp4 8-6 redis sentinel安装演示-2.mp4 8-5 redis sentinel安装演示-1.mp4 8-4 redis sentinel安装与配置.mp4 8-3 redis sentinel架构.mp4 8-2 主从复制高可用?.mp4 8-19 本章总结.mp4 8-18 高可用读写分离.mp4 8-17 节点运维.mp4 8-16 常见开发运维问题-目录.mp4 8-15 故障转移.mp4 8-14 领导者选举.mp4 8-13 主观下线和客观下线.mp4 8-12 三个定时任务.mp4 8-11 实现原理-3.故障演练(日志分析).mp4 8-10 实现原理-2.故障转移演练(客户端).mp4 8-1 sentinel-目录.mp4 7-9 主从复制常见问题.mp4 7-8 故障处理.mp4 7-7 全量复制开销 + 部分复制.mp4 7-6 全量复制.mp4 7-5 runid和复制偏移量.mp4 7-4 主从复制配置-操作.mp4 7-3 主从复制配置-介绍.mp4 7-2 什么是主从复制.mp4 7-1 目录.mp4 6-4 AOF阻塞.mp4 6-3 子进程开销和优化.mp4 6-2 fork.mp4 6-1 常见问题目录.mp4 5-9 RDB和AOF抉择.mp4 5-8 AOF实验.mp4 5-7 AOF(2).mp4 5-6 AOF(1).mp4 5-5 RDB(3).mp4 5-4 RDB(2).mp4 5-3 RDB(1).mp4 5-2 持久化的作用.mp4 5-1 目录.mp4 4-7 geo.mp4 4-6 hyperloglog.mp4 4-5 bitmap.mp4 4-4 发布订阅.mp4 4-3 pipeline.mp4 4-2 慢查询.mp4 4-1 课程目录.mp4 3-4 Go客户端:redigo简介.mp4 3-3 Python客户端:redis-py.mp4 3-2 Java客户端:Jedis.mp4 3-1 课程目录.mp4 2-9 list(2).mp4 2-8 list(1).mp4 2-7 hash (2).mp4 2-6 hash (1).mp4 2-5 字符串.mp4 2-4 单线程.mp4 2-3 数据结构和内部编码.mp4 2-2通用命令.mp4 2-11 zset.mp4 2-10 set.mp4 13-1 _课程总结.mp4 12-7 运维功能.mp4 12-6 用户功能.mp4 12-5 应用接入.mp4 12-4 机器部署.mp4 12-3 _快速构建.mp4 12-2 _Redis规模化困扰.mp4 12-1 _目录.mp4 11-8 本章总结.mp4 11-7 热点key的重建优化.mp4 11-6 无底洞问题.mp4 11-5 缓存穿透问题.mp4 11-4 缓存粒度问题.mp4 11-3 缓存的更新策略.mp4 11-2 缓存的受益和成本.mp4 11-1 目录.mp4 10-9 集群缩容-操作.mp4 10-8 集群缩容-说明.mp4 10-7 集群扩容演示-2.mp4 10-6 集群扩容演示-1.mp4 10-5 扩展集群-3.迁移槽和数据.mp4 10-4 扩展集群-2.加入集群.mp4 10-35 本章总结.mp4 10-34 集群vs单机.mp4 10-33 数据迁移.mp4 10-32 读写分离.mp4 10-31 请求倾斜.mp4 10-30 数据倾斜.mp4 10-3 扩展集群-1.加入节点.mp4 10-29 集群倾斜-目录.mp4 10-28 PubSub广播.mp4 10-27 带宽消耗.mp4 10-26 集群完整性.mp4 10-25 Redis Cluster常见开发运维问题-目录.mp4 10-24 故障模拟.mp4 10-23 故障恢复.mp4 10-22 故障发现.mp4 10-21 故障转移-目录.mp4 10-20 批量操作优化.mp4 10-2 集群伸缩原理.mp4 10-19 多节点操作命令.mp4 10-18 整合spring-2.mp4 10-17 整合spring-1.mp4 10-16 JedisCluster基本使用.mp4 10-15 smart客户端JedisCluster-目录.mp4 10-14 JedisCluster执行源码分析.mp4 10-13 smart客户端实现原理.mp4 10-12 ask重定向.mp4 10-11 moved异常说明和操作.mp4 10-10 客户端路由-目录.mp4 10-1 集群伸缩目录.mp4 1-9 特性5-功能丰富.mp4 1-8 特性4-多语言客户端.mp4 1-7 特性3-数据结构.mp4 1-6 特性2-持久化.mp4 1-5 特性1-速度快.mp4 1-4 redis特性目录.mp4 1-3 谁在使用Redis.mp4 1-2 Redis初识.mp4 1-15 redis常用配置.mp4 1-14 redis三种启动方式介绍.mp4 1-13 redis典型使用场景.mp4 1-12 特性8-高可用分布式.mp4 1-11 特性7-复制.mp4 1-10 特性6-简单.mp4 1-1 导学.mp4
Redis可以用来作为分布式缓存分布式锁和消息队列的原因有以下几点。 首先,Redis具有高性能的内存数据库特性,在缓存场景下,能够实现快速的读写操作,提高数据的访问速度。通过将常用的数据存储在Redis中,可以避免频繁地查询数据库,减少响应时间,提高系统的吞吐量。 其次,Redis支持数据过期机制和LRU(Least Recently Used,最近最少使用)算法等缓存策略,可以根据需求定时清理过期数据或者根据缓存数据的使用频率进行淘汰,保证缓存的有效性和一定的容量。 此外,Redis还提供了常见的数据结构,例如字符串、列表、哈希、集合和有序集合等,这些数据结构的操作都是原子性的,可以支持多个客户端同时访问和修改数据,利于实现分布式锁功能。通过Redis的SETNX(SET if Not eXists)指令可以实现简单的互斥锁机制,通过设定过期时间和唯一标识可以防止死锁的发生。 此外,通过Redis的订阅与发布机制,可以轻松地实现消息队列的功能。发布者可以发布消息,订阅者可以实时地接收到消息并进行相应处理。同时,Redis还支持发布与订阅模式的消息持久化,即使在消息发布者和订阅者之间存在断开连接的情况下,消息也不会丢失。 综上所述,Redis的高性能、灵活的缓存策略、原子性的数据操作和消息持久化机制,使其成为一个强大的工具,可以用来实现分布式缓存分布式锁和消息队列的功能

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

taoli-qiao

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

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

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

打赏作者

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

抵扣说明:

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

余额充值