SpringBoot之使用Redisson实现分布式锁(含完整例子)

SpringBoot之使用Redisson实现分布式锁(含完整例子)

  前言:分布式环境有很多问题,比如你前一个请求访问的是服务器A,第二个请求访问到了服务器B,就会发生并发重复插入问题,这个时候需要依赖单点来实现分布锁,而redis就是。
  Redisson的分布式锁的实现,一般提及到Redis的分布式锁我们更多的使用的是Redisson的分布式锁,Redis的官方也是建议我们这样去做的。Redisson点我可以直接跳转到Redisson的官方文档

  1.引入Maven依赖

<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson-spring-boot-starter</artifactId>
    <version>3.10.6</version>
</dependency>

注意:我这里引入的是redisson和springboot的集成包,网上一些教程可能是引入如下配置

<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.6.1</version>
</dependency>

  1.2 配置redis信息


spring:
  redis:
    host: ${REDIS_URL:192.168.6.52}
    port: ${REDIS_PORT:6379}
    password: ${REDIS_PWD:123456}

  1.3 配置redisson
在这里插入图片描述
  新建一个redisson-single.yml的配置文件 下面是单机配置


singleServerConfig:
  idleConnectionTimeout: 10000
  pingTimeout: 1000
  connectTimeout: 10000
  timeout: 3000
  retryAttempts: 3
  retryInterval: 1500
  reconnectionTimeout: 3000
  failedAttempts: 3
  password: 123456
  subscriptionsPerConnection: 5
  clientName: null
  address: "redis://192.168.6.52:6379"
  subscriptionConnectionMinimumIdleSize: 1
  subscriptionConnectionPoolSize: 50
  connectionMinimumIdleSize: 32
  connectionPoolSize: 64
  database: 0
  #在最新版本中dns的检查操作会直接报错 所以我直接注释掉了
  #dnsMonitoring: false
  dnsMonitoringInterval: 5000
threads: 0
nettyThreads: 0
codec: !<org.redisson.codec.JsonJacksonCodec> {}

  1.4 写一个RedissonConfig配置类 来配置你的redisson

/**
 * @author tbb
 *
 */
@Configuration
public class RedssonConfig
{
	@Bean(destroyMethod = "shutdown")
	public RedissonClient redisson() throws IOException
	{
		RedissonClient redisson = Redisson.create(Config.fromYAML(new ClassPathResource("redisson-single.yml").getInputStream()));
		return redisson;
	}
}

或者使用如下配置类

/**
 * @author tbb
 *
 */
@Configuration
public class RedssonConfig
{
	    @Bean
	    public RedissonClient getRedisson() 
	    {
	        Config config = new Config();
	        config.useSingleServer()
	                .setAddress("redis://192.168.6.52:6379").setPassword("123456")
	                .setRetryInterval(5000)
	                .setTimeout(10000)
	                .setConnectTimeout(10000);
	        return Redisson.create(config);
	    }



}

  1.5 写一个简单的测试请求

	@ApiOperation("分布式锁测试")
    @ResponseBody
	@GetMapping("test")
	public String createOrderTest() 
	{
	    return entityService.decreasePrice("1234-1234-1234", 1);
	}

  1.6 编写一个砍价接口

@Service
@Transactional(readOnly = true, rollbackFor = Exception.class)
public class ShwzTestService extends CrudService<ShwzTestDao, ShwzTest, String>
{

	@Autowired
	private RedissonClient redissonClient;

	public String decreasePrice(String productId, Integer productQuantity)
	{
		String key = "dec_store_lock_" + productId;
		RLock lock = redissonClient.getLock(key);
		ShwzTest entity = new ShwzTest();
		entity.setId(productId);
		ShwzTest shwzTest = this.get(entity);
		lock.lock();
		// 加锁 操作很类似Java的ReentrantLock机制
		try
		{
			
			// 简单减价操作 没有重新写其他接口了
			if (shwzTest.getPrice() == 0)
			{
				return "当前价格为0,不能在减了!!";
			}
			shwzTest.setPrice(shwzTest.getPrice() - 1);
			this.save(shwzTest);
		}
		catch (Exception e)
		{
			System.out.println(e.getMessage());
		}
		finally
		{
			// 解锁
			lock.unlock();
		}
		String result = "当前价格:" + shwzTest.getPrice();
		return result;
	}
}

1.7、使用ab做接口测试

ab -c 1000 -n 1000 -H "Authorization:bearer46a2b87f-0f8a-477d-bcaf-dad144860a23" http://192.168.0.6:7001/a/shwz/shwzTest/test

在这里插入图片描述

ab -n 300 -c 300 请求地址
-n 的含义就是你做多少个请求
-c 的含义就是多少个用户并发请求
-H  的含义就是请求的Header头文件

18、测试完成 price 价格减到0 之后就没有在减少了
在这里插入图片描述

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值