Spring boot 整合 Redisson实现分布式锁并验证功能

 更新时间:2022年05月19日 10:48:59   作者:幻夜星辰  
这篇文章主要介绍了Spring boot 整合 Redisson实现分布式锁并验证,redisson是官方推荐的分布式锁实现方案,采用redis自身的原子命令和lua脚本来实现,需要的朋友可以参考下

简述

整篇文章写的比较粗糙,大佬看了轻喷。前半部分 是整合spring boot和redisson, 后半部分是验证分布式锁。在整个过程中遇见了不少的问题,在此做个记录少走弯路

redisson是官方推荐的分布式锁实现方案,采用redis自身的原子命令和lua脚本来实现

1. 在idea中新建spring boot工程并引入所需依赖

idea中直接新建一个spring boot项目即可,再在pom.xml中引入所需依赖,依赖信息如下

  <!-- redis所需 -->
         <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>
        <!-- web页面访问所需 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson-spring-data-25</artifactId>
    <version>3.16.4</version>
</dependency>
<!-- Redisson所需依赖 -->
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson-spring-boot-starter</artifactId>
    <version>3.16.4</version>
</dependency>

2. 编写相关代码实现

采用一个票卷库存来进行锁的验证,需要预先在redis里面设置一个key为ticket的信息,值为100之类的数字即可 代码示例编写:

@RestController
@Slf4j
public class RedisController {
    // spring boot 操作redis的模板方法类
    @Autowired
    private StringRedisTemplate redisTemplate;

    // redisson操作bean
    @Resource
    private Redisson redisson;
    @RequestMapping("/lock")
    public String deductTicket(){
        String lockKey = "ticket";
        
        // 在spring boot 2.0.6版本中整合的redisson,key和锁不能一样
        // redis setnx 操作,此处的lockKey在后面追加1是为了避免redisson锁时报错, 需要和待锁住的数据的key信息不同
        RLock lock = redisson.getLock(lockKey+"1");
        try {
            lock.lock();
            int ticketCount = Integer.parseInt(redisTemplate.opsForValue().get(lockKey));
            if (ticketCount > 0) {
                int realTicketCount = ticketCount - 1;
                log.info("扣除成功:剩余票数:" + realTicketCount);
                redisTemplate.opsForValue().set(lockKey, realTicketCount + "");
                return realTicketCount + "";
            } else {
                log.error("扣除失败");
                return "error";
            }
            return "end";
        } finally {
            lock.unlock();
        }
    }
}

application.yml配置信息如下

server:
  port: 8899  # web服务对外端口
  redis:
    host: 192.168.0.160 #redis地址
    database: 0 #采用的库编号
    port: 6379 #redis端口
    password: 123456 #redis密码,如果redis没有设置密码直接去掉该配置不写空

3. 模拟实际环境验证

一个简单的分布式锁验证的demo完成了,比较粗糙,验证的方式有多样的,可以采用java本身的多线程进行验证,也可以类似实际环境部署多个节点来验证,这里为了技术的广度的一个应用,采用后面的方式。

3.1 下载idea的docker插件并配置相关镜像信息

  • docker插件下载

  • idea配置docker连接虚拟机上的docker

    开启docker远程连接

vim /usr/lib/systemd/system/docker.service

找到 ExecStart,在最后面添加 -H tcp://0.0.0.0:2375

重启docker

systemctl daemon-reload 
systemctl start docker

开放端口

firewall-cmd --zone=public --add-port=2375/tcp --permanent

idea配置docker连接

配置完成后可在idea中看到连接的docker镜像、容器相关信息

3.2 将spring boot打包的jar构建为docker镜像

编写Dockerfile

在工程目录中新建Dockerfile,与pom.xml文件同级

Dockerfile内容如下

FROM openjdk:8-jdk-alpine # 指定基础镜像为jdk8
ADD target/spring-0.0.1-SNAPSHOT.jar app.jar #将打包的jar包放入镜像中并重命名为app.jar
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/app.jar"]  #启动jar包

在pom.xml中添加插件信息

<plugin>
    <groupId>com.spotify</groupId>
    <artifactId>docker-maven-plugin</artifactId>
    <version>1.0.0</version>
    <configuration>
        <imageName>com.demo/${project.artifactId}</imageName>
        <dockerDirectory></dockerDirectory>
        <resources>
            <resource>
                <targetPath>/</targetPath>
                <directory>${project.build.directory}</directory>
                <include>${project.build.finalName}.jar</include>
            </resource>
        </resources>
    </configuration>
</plugin>

idea添加docker构建配置

此处操作的目的是为了便于在idea中直接构建镜像启动容器

配置信息如下

(1)Dockerfile文件位置

(2)镜像tag

(3)构建成功后启动的容器名称

(4)端口映射 宿主机端口:容器内端口

jar包打好之后可直接点击

如果没有问题idea的docker控制台会输出容器内启动相关日志信息

此处为了验证分布式锁,需要两个节点以上,所以这里我手动启动了另外一个docker容器(暂时没有编写shell脚本)

启动命令如下

docker run -d --name demo2 -p 8900:8899 demo:1.1

容器名称demo2 映射到宿主机端口 8900

查看应用日志可使用 docker logs -f 容器名称

spring boot 构建docker镜像应该有更简单的方式,欢迎在评论区交流补充

3.2 配置nginx

可以更改默认的日志格式如下,为了记录代理的具体节点信息

 log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for" ' '"$upstream_addr"';

配置负载均衡

upstream tomcat { 
        server 192.168.0.160:8899 weight=10; # 此处设置权重为相同的即可
        server 192.168.0.160:8900 weight=10;
    }

配置代理

location  / {
           proxy_pass http://tomcat;
           proxy_redirect default;
        }

完整的代理配置如下

配置完成后

nginx -s reload

进行访问验证

192.168.0.160/lock

可查看nginx、容器内日志信息来验证是否访问成功

nginx如何访问出现error,可查看nginx中的错误日志。如果是权限问题(# failed (13: Permission denied) while connecting to upstream) 可用root用户启动或使用命令 setsebool -P httpd_can_network_connect 1 来解决

3.3 下载安装Jmeter进行测试

Jmeter下载地址:jmeter.apache.org/download_jm…

下载解压后在bin目录中店家jmeter.bat即可启动jmeter

新增计划

配置线程信息

(1) 请求线程数

(2) 多少s内启动完线程 设置为0代表同时启动 设置为2代表2s内启动完20个线程,1s启动10个线程

添加请求

(1) ip地址

(2) 端口

(3) path信息,这里设置为jar中的地址信息

点击绿色三角箭头启动并进行验证

查看redis缓存中设置的数据是否在测试完成后符合预期值,也可查看nginx中的日志来确定请求是否平均分配到了两个节点

参考文章:
juejin.cn/post/684490…

juejin.cn/post/709110…

到此这篇关于Spring boot 整合 Redisson实现分布式锁并验证的文章就介绍到这了,更多相关Spring boot 分布式锁内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • feign服务端发现异常客户端处理的方法介绍

    feign服务端发现异常客户端处理的方法介绍

    这篇文章主要给大家介绍了关于feign服务端发现异常客户端处理的方法,文中通过示例代码介绍的非常详细,对大家学习或者使用feign具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-07-07
  • 关于fastjson的常见API详解

    关于fastjson的常见API详解

    这篇文章主要介绍了关于fastjson的常见API详解,Fastjson是一个Java库,可用于将Java对象转换为其JSON表示,它还可用于将JSON字符串转换为等效的Java对象,Fastjson可以处理任意Java对象,包括您没有源代码的预先存在的对象,需要的朋友可以参考下
    2023-07-07
  • SpringBoot整合Flink CDC实现实时追踪mysql数据变动

    SpringBoot整合Flink CDC实现实时追踪mysql数据变动

    我们将整合Spring Boot和Apache Flink CDC(Change Data Capture)来实现实时数据追踪,下面是一个基本的实践流程代码,包括搭建Spring Boot项目、整合Flink CDC以及实现数据变动的实时追踪,需要的朋友可以参考下
    2024-07-07
  • Java中获取Class对象的3种方式代码示例

    Java中获取Class对象的3种方式代码示例

    Class对象是反射的核心,通过他可以调用类的任意方法,下面这篇文章主要给大家介绍了关于Java中获取Class对象的3种方式,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-07-07
  • 如何通过Java监听MySQL数据的变化

    如何通过Java监听MySQL数据的变化

    对于二次开发来说,很大一部分就找找文件和找数据库的变化情况,下面这篇文章主要给大家介绍了关于如何通过Java监听MySQL数据的变化的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-03-03
  • mybatis查询SqlServer慢问题及解决

    mybatis查询SqlServer慢问题及解决

    这篇文章主要介绍了mybatis查询SqlServer慢问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • java中Map遍历的四种方式总结

    java中Map遍历的四种方式总结

    Map集合遍历日常开发最常使用,下面这篇文章主要给大家总结介绍了关于java中Map遍历的四种方式,文中通过实例代码介绍的非常详细,对大家学习或者使用java具有一定的参考学习价值,需要的朋友可以参考下
    2023-03-03
  • 基于Jackson实现API接口数据脱敏的示例详解

    基于Jackson实现API接口数据脱敏的示例详解

    用户的一些敏感数据,例如手机号、邮箱、身份证等信息,在数据库以明文存储,但在接口返回数据给浏览器(或三方客户端)时,希望对这些敏感数据进行脱敏,所以本文就给大家介绍以恶如何利用Jackson实现API接口数据脱敏,需要的朋友可以参考下
    2023-08-08
  • Jmeter BlazeMeter实现web录制过程

    Jmeter BlazeMeter实现web录制过程

    BlazeMeter是一款与Apache JMeter兼容的chrome插件,采用BlazeMeter可以方便的进行流量录制和脚本生成,作为接口测试脚本编写的一个基础,这篇文章主要介绍了Jmeter BlazeMeter实现web录制,需要的朋友可以参考下
    2021-12-12
  • Java OpenCV实现人脸识别过程详解

    Java OpenCV实现人脸识别过程详解

    这篇文章主要介绍了Java OpenCV实现人脸识别过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-08-08

最新评论