springboot+redisson+nacos

使用nacos配置管理,实现redisson分布式锁

添加依赖

 <dependency>
            <groupId>org.redisson</groupId>
            <artifactId>redisson-spring-boot-starter</artifactId>
            <version>3.16.0</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba.boot</groupId>
            <artifactId>nacos-config-spring-boot-starter</artifactId>
            <version>0.2.1</version>
        </dependency>

启动类配置nacos

@SpringBootApplication
@NacosPropertySource(dataId = "test",groupId = "DEFAULT_GROUP",autoRefreshed = true)
public class SprApplication {

    public static void main(String[] args) {
        SpringApplication.run(SprApplication.class, args);
    }


}

redisson配置

@Bean(name = "redissonClient")
   public RedissonClient redissonClient() {
       Config config = new Config();
       config.setLockWatchdogTimeout(30000);
       config.setCodec(new StringCodec());
       SingleServerConfig singleServerConfig = config.useSingleServer().setAddress(address);
       singleServerConfig.setPassword(this.getPassword());
       singleServerConfig.setConnectionPoolSize(connectionPoolSize);
       System.out.println(("RedissonClient init success!")+singleServerConfig.getAddress());

       RedissonClient redissonClient = Redisson.create(config);

       return redissonClient;
   }

redis配置

  @Bean(name = "redisConnectionFactory")
    public RedisConnectionFactory redisConnectionFactory() {
        JedisConnectionFactory factory = new JedisConnectionFactory();
        factory.setHostName(this.getHost());
        factory.setUsePool(true);
        factory.setPort(this.getPort());
        factory.setTimeout(10000);
        factory.setPassword(this.password);
        JedisPoolConfig config = (JedisPoolConfig) factory.getPoolConfig();
        config.setMaxTotal(10000);
        config.setMaxIdle(1000);
        return factory;
    }
    @Bean
    public JedisPoolConfig jedisPoolConfig() {
        JedisPoolConfig poolConfig = new JedisPoolConfig();
        poolConfig.setMaxIdle(100);
        poolConfig.setMinIdle(100);
        poolConfig.setTestOnCreate(true);
        poolConfig.setTestOnBorrow(true);
        poolConfig.setTestOnReturn(true);
        poolConfig.setTestWhileIdle(true);
        long maxWaitMillis = 1000 * 60 * 5;
        poolConfig.setMaxWaitMillis(maxWaitMillis);
        return poolConfig;
    }
    @Bean("redisTemplate")
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
        RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
        redisTemplate.setKeySerializer(new StringRedisSerializer());
        redisTemplate.setHashKeySerializer(new StringRedisSerializer());
        redisTemplate.setHashValueSerializer(new StringRedisSerializer());
        redisTemplate.setValueSerializer(new StringRedisSerializer());
        redisTemplate.setConnectionFactory(factory);

        return redisTemplate;
    }

线程池的大小根据自己具体情况确定。

可重入锁实现使用

创建Redisson客户端对象。
使用Redisson客户端对象获取一个可重入锁(ReentrantLock)。
在需要使用分布式锁的代码段中,先调用lock()方法获取锁。
执行业务逻辑。
执行完业务逻辑后,调用unlock()方法释放锁。
如果在获取锁的过程中发生了异常,需要调用unlock()方法来释放锁。

 public void myMethod() {
        String key = "111";
        String lockKey = "lock:" + key;
        // 可重入锁
        RLock lock = redissonClient.getLock(lockKey);
        boolean isLocked = lock.tryLock();
        System.out.println(isLocked);
        if (isLocked) {
            try {
                System.out.println("Lock");
            } catch (Exception e) {
                System.out.println(e);
            } finally {
                if (lock.isHeldByCurrentThread()) {
                    lock.unlock();
                }
            }
        }

    }

公平锁

Redisson公平锁(FairLock)是一种基于Redis实现的分布式公平锁,它可以保证获取锁的顺序和请求锁的顺序一致,避免“饥饿”现象的发生。

在Redisson中,使用公平锁的方式与可重入锁类似。通过Redisson客户端对象获取公平锁对象,然后调用lock()方法获取锁,在获取锁时,根据请求锁的顺序依次排队等待锁的释放,当锁被释放后,等待时间最长的线程会最先获得锁。

RedissonClient redissonClient = Redisson.create();
RLock fairLock = redissonClient.getFairLock("myFairLock");

// 获取锁
fairLock.lock();

try {
    // 执行业务逻辑
} finally {
    // 释放锁
    fairLock.unlock();
}

读写锁(ReadWriteLock)

是一种用于多线程环境下对共享资源进行读写操作的锁,它可以提高系统的并发性能和可伸缩性,适用于读操作频繁、写操作较少的场景。

在读写锁中,读操作可以并发进行,写操作需要独占锁资源。当有一个线程获取写锁时,其他线程不能获取读锁或写锁,只有该线程释放锁之后,其他线程才能获取锁。读写锁的使用可以有效地避免读写冲突问题,提高系统的并发性能和响应速度。

Redisson也提供了读写锁的实现,可以使用RLock readWriteLock = redissonClient.getReadWriteLock("myReadWriteLock")获取读写锁对象,并使用readLock()方法获取读锁对象、writeLock()方法获取写锁对象,具体示例代码如下

RedissonClient redissonClient = Redisson.create();
RLock readWriteLock = redissonClient.getReadWriteLock("myReadWriteLock");

// 获取读锁
RLock readLock = readWriteLock.readLock();
readLock.lock();

try {
    // 读取共享资源的数据
} finally {
    readLock.unlock();
}

// 获取写锁
RLock writeLock = readWriteLock.writeLock();
writeLock.lock();

try {
    // 修改共享资源的数据
} finally {
    writeLock.unlock();
}

限流器(RateLimiter)

是一种用于限制系统流量的工具,它可以控制系统处理请求的速率,避免系统因为过多的请求导致的性能下降或者宕机的问题。常见的限流算法有令牌桶算法和漏桶算法等。

Redisson提供了令牌桶算法的实现,可以使用RTopic.create(redissonClient).getBucket("myBucket").trySetRate(RateType type, long rate, long rateInterval, long rateIntervalUnit)方法创建一个令牌桶,并指定限流速率、令牌桶容量等参数。其中,rate参数表示每秒生成的令牌数量,rateInterval和rateIntervalUnit参数表示生成令牌的时间间隔。

在使用限流器时,需要在处理请求之前先从限流器中获取令牌,如果获取成功,则可以处理请求,否则需要等待一段时间再重试。Redisson的令牌桶限流器的示例代码如下:

RedissonClient redissonClient = Redisson.create();

// 创建一个每秒生成10个令牌的令牌桶
RateLimiter rateLimiter = redissonClient.getRateLimiter("myRateLimiter");
rateLimiter.trySetRate(RateType.OVERALL, 10, 1, TimeUnit.SECONDS);

// 处理请求
while (true) {
    // 判断是否能够获取到令牌,获取不到则等待一段时间再重试
    if (rateLimiter.tryAcquire()) {
        // 处理请求
        System.out.println("处理请求");
    } else {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}


一般来说,被限流器限流的请求应该返回一个提示,告知用户请求被限流了。具体的提示信息可以根据业务需求进行定义,例如可以返回一个HTTP状态码为429的响应,并在响应消息中包含“请求过于频繁,请稍后再试”等提示信息。
对于被限流的请求,除了返回一个提示信息外,还可以根据实际情况选择不同的处理方式。例如可以将被限流的请求放入队列中,等待限流器解除限制后再进行处理;或者将被限流的请求进行缓存,等待一定时间后再进行处理。在实际应用中,需要根据业务需求和系统架构进行选择和调整,以实现最佳的流量控制效果。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容