首页 > 数据库 >使用StringRedisTemplate实现redis分布式锁

使用StringRedisTemplate实现redis分布式锁

时间:2023-02-28 22:36:44浏览次数:54  
标签:spring boot redis springframework import org StringRedisTemplate 分布式

背景:单个接口可能同时被多个用户调用,但是每个用户使用的数据都是不一样,因此需要使用分布式锁解决数据减少了没有即使减少的问题

使用的指令来自的edis的setnx命令,setnx(k,v1),setnx(k.v2),当设置值为v1后,v2的设置无效

 

 

 上图中启动了两台一样的服务,大多数情况,同一时间只有一条日志打印,但是定时任务是每15秒执行一次,数据有效期15秒,所以会有同一秒不同毫秒的日志打印

 当时间设置为

timed.task=0 30,35,40 17 * * *

  

 

 同一秒内无法保证只执行一次

 

 但是到了毫秒级别,不一样,实际上是不一样用户,但是定时任务部署多台服务器,这样可能存在问题

 

上图看到没有了在一秒内重合的日志打印,这边是调整了锁的过期时间

 

 

该行代码保证了原子性
Boolean setIfAbsent = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, "001", 25, TimeUnit.SECONDS);

相关代码部分

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.redis</groupId>
    <artifactId>redis-service</artifactId>
    <version>1.0-SNAPSHOT</version>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.1.RELEASE</version>
        <relativePath/>
    </parent>

    <dependencies>
        <!--tomcat容器-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>
        <!--lombok依赖-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.16</version>
        </dependency>

        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-pool2</artifactId>
            <version>2.4.1</version>
        </dependency>
    </dependencies>


</project>

  

package com.redis;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2023/2/28 14:38
 */
@SpringBootApplication
@EnableAsync
public class RedisApplication {
    public static void main(String[] args) {

        SpringApplication.run(RedisApplication.class, args);

    }
}

  

package com.redis.task;

import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.scheduling.annotation.Async;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.Scheduled;

import java.util.UUID;
import java.util.concurrent.TimeUnit;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2023/2/28 14:51
 */
@Configuration      //1.主要用于标记配置类,兼备Component的效果。
@EnableScheduling   // 2.开启定时任务
@Slf4j
public class RedisTimeTask {


    @Autowired
    private StringRedisTemplate stringRedisTemplate;


    @Scheduled(cron = "${timed.task}")//每天上午9点,下午5点执行

    /**
     * 单位都是毫秒
     */
   // @Scheduled(initialDelay = 5000,fixedRate=5000)

   // @Scheduled(initialDelay = 5000,fixedRate=20000)
    @Async
    public void timerSendMessage(){
        String lockKey="test_001";
//        Boolean setIfAbsent = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, "001");
//        stringRedisTemplate.expire(lockKey,25, TimeUnit.SECONDS);

        Boolean setIfAbsent = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, "001", 25, TimeUnit.SECONDS);

        if (setIfAbsent){

            try {

                log.info("测试redis的分布式锁成功....");

            } finally {
               stringRedisTemplate.delete(lockKey);
            }
        }

    }
}

  

server.port=8081

# 当天17点25分0秒
#timed.task=0 05,10,15,20,25,30,35,40,45,50,55 * * * *
#每隔15秒执行一次
timed.task=0/15 * * * * *

#timed.task= 0 0/5 * * * *

#redis配置
spring.redis.host=127.0.0.1
spring.redis.port=6380
spring.redis.database=1
spring.redis.jedis.pool.max-active=8
spring.redis.jedis.pool.min-idle=0
spring.redis.password=
spring.redis.lettuce.pool.max-wait=1000

logging.level.com.redis=debug
logging.level.web=debug
spring.devtools.add-properties=false

  

 

标签:spring,boot,redis,springframework,import,org,StringRedisTemplate,分布式
From: https://www.cnblogs.com/q202105271618/p/17162279.html

相关文章

  • 分布式锁 -- redis
    原理redis设置一个key和value,如果存在则获取锁失败,不存在则获取锁成功处理业务,业务处理完成后删除这条数据,可以带个失效时间。 代码publicvoidhandleInvoice(Subm......
  • Rocky Linux 9 安装 Redis 6 哨兵配置
    1、安装Redis6.2(四个节点,一主两副本、一哨兵)#所有节点都安装dnf-yinstallredis2、修改主节点配置#修改部分的配置文件#vim/etc/redis/redis.conf...bin......
  • redis(12)持久化操作-RDB
    前言Redis提供了2个不同形式的持久化方式:RDB(RedisDataBase)AOF(AppendOfFile) RDB在指定的时间间隔内将内存中的数据集快照写入磁盘,也就是行话讲的Snaps......
  • 高可用架构设计ES+Redis+MySQL
    1、项目背景会员系统是一种基础系统,跟公司所有业务线的下单主流程密切相关。如果会员系统出故障,会导致用户无法下单,影响fanwei是全公司所有业务线。所以,会员系统必须保证......
  • springboot+mybatis+redis+mysql项目搭建,含示例Demo
    转载自:https://blog.csdn.net/qq_40772342/article/details/105049322========== redis在web开发中使用的场景很多,其中缓存是其中一个很重要的使用场景,之所以用作缓存,......
  • 开源分布式任务调度系统就选:DolphinScheduler
    分布式任务调度这个话题是每个后端开发和大数据开发都会接触的话题。因为应用场景的广泛,所以有很多开源项目专注于解决这类问题,比如我们熟知的xxl-job。那么今天要给大家......
  • PHP基于Redis实现延迟队列
    PHP基于Redis实现延迟队列标签(空格分隔):php,redis,linux解决思路1.基于Redis过期key事件2.基于RedisZset特征处理Redis-ZsetZset首先它是一个set,这保证了内部......
  • spring redis 工具类
    /***springredis工具类**@authorhanzj**///@SuppressWarnings(value={"unchecked","rawtypes"})@ServicepublicclassRedisForLbCache{@Autowiredp......
  • 《分布式技术原理与算法解析》学习笔记Day25
    负载均衡负载均衡是分布式可靠性中非常关键的一个问题,它在一定程度上反映了分布式系统对业务处理的能力。什么是负载均衡?负载均衡可以分为两种:请求负载均衡,即将用户的......
  • redis的安装
    https://github.com/tporadowski/redis/releases下载地址解压到D:\redis目录,配置环境变量,设置REDIS_HOME值为解压的目录,在path里面引用该HOME,然后在cmd输入redis-cli.exe,......