当前位置: 首页 > news >正文

动易论坛官方网站2w网站2w网站建设建设

动易论坛官方网站,2w网站2w网站建设建设,建设学院2级网站的作用,企业年报系统官网入口本博客为个人学习笔记#xff0c;学习网站与详细见#xff1a;黑马程序员Redis入门到实战 P56 - P63 目录 分布式锁介绍 基于Redis的分布式锁 Redis锁代码实现 修改业务代码 分布式锁误删问题 分布式锁原子性问题 Lua脚本 编写脚本 代码优化 总结 分布式锁介绍…  本博客为个人学习笔记学习网站与详细见黑马程序员Redis入门到实战 P56 - P63 目录 分布式锁介绍 基于Redis的分布式锁 Redis锁代码实现 修改业务代码  分布式锁误删问题 分布式锁原子性问题  Lua脚本 编写脚本  代码优化 总结  分布式锁介绍 在上一篇文章 Redis实战—优惠卷秒杀 中我们通过使用锁、事务和代理对象实现了“一人一单”的优惠券秒杀功能。但我们使用的锁是基于JVM内部的锁这导致锁的范围只能限制单个JVM的线程操作因此在集群情况下依然会出现超卖问题。所以我们需要设置一个锁使其能够同时限制集群中的多个JVM线程操作而这个锁就是分布式锁由此引出本文。 集群情况下JVM锁的使用情况如下图。 集群情况下分布式锁的使用情况如下图。 分布式锁的实现 基于Redis的分布式锁 我们利用Redis的SET lock thread1 NX操作来模拟获取锁即如果当前不存在lock键则添加lock键成功如果当前存在lock键则添加lock键失败。我们将添加lock键的操作视为获取锁的操作将lock键是否存在视为当前锁是否已被其他线程获取。执行语句后通过Redis返回OK或者nil我们可以判断是否获取锁成功。为防止宕机时无法对锁进行销毁我们在进行SET操作时还需通过EX为键设置一个合理的时间。 Redis锁代码实现 // 接口类 public interface ILock {/** 尝试获取锁* timeoutSec 锁持有的超时时间过期后自动释放* 返回值 true代表获取锁成功false代表获取锁失败* */boolean tryLock(long timeoutSec);//释放锁void unlock();}// 接口实现类 public class SimpleRedisLock implements ILock {private String name;private StringRedisTemplate stringRedisTemplate;public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {this.name name;this.stringRedisTemplate stringRedisTemplate;}private static final String KEY_PREFIX lock:;Overridepublic boolean tryLock(long timeoutSec) {// 获取线程标识long threadId Thread.currentThread().getId();// 获取锁并添加时间Boolean success stringRedisTemplate.opsForValue().setIfAbsent(KEY_PREFIX name, threadId , timeoutSec, TimeUnit.SECONDS);//避免拆箱导致空指针使用Boolean.TRUE.equals方法返回结果return Boolean.TRUE.equals(success);}Overridepublic void unlock() {// 释放锁stringRedisTemplate.delete(KEY_PREFIX name);} }修改业务代码  public Result seckillVoucher(Long voucherId) {//判断是否满足抢购条件...Long userId UserHolder.getUser().getId();// 创建锁对象根据用户ID加锁SimpleRedisLock lock new SimpleRedisLock(order: userId, stringRedisTemplate);// 获取锁boolean isLock lock.tryLock(1200);// 若获取锁失败if (!isLock)return Result.fail(不允许重复下单);// 若获取锁成功try {// 获取当前代理对象事务IVoucherOrderService proxy (IVoucherOrderService) AopContext.currentProxy();return proxy.createVoucherOrder(voucherId);} finally {lock.unlock();}} 分布式锁误删问题 如上图所示持有锁的线程1在锁的内部出现了业务阻塞导致它的锁被超时释放。这时线程2尝试获得锁成功然而在线程2持有锁执行过程中线程1的业务反应过来继续执行而线程1业务执行完成后进行了删除锁逻辑此时就会把本应属于线程2的锁进行删除这就是误删其它线程锁的情况。  解决方案当线程创建锁时同时为该锁添加当前线程标识该标识由UUID随机数为前缀与线程id组合而成为避免出现集群下两个线程的id相同的情况因此添加UUID前缀。当一个线程删除锁时需要判断当前线程标识与锁标识是否一致若一致说明该锁由当前线程创建可进行删除若不一致说明该锁由其它线程创建不可进行删除。 对simpleRedisLock类代码优化如下。 package com.hmdp.utils;import cn.hutool.core.lang.UUID; import org.springframework.data.redis.core.StringRedisTemplate;import java.util.concurrent.TimeUnit;public class SimpleRedisLock implements ILock {private String name;private StringRedisTemplate stringRedisTemplate;public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {this.name name;this.stringRedisTemplate stringRedisTemplate;}private static final String KEY_PREFIX lock:;private static final String ID_PREFIX UUID.randomUUID().toString(true) -;Overridepublic boolean tryLock(long timeoutSec) {// 获取线程标识String threadId ID_PREFIX Thread.currentThread().getId();// 获取锁并设置标识、添加时间Boolean success stringRedisTemplate.opsForValue().setIfAbsent(KEY_PREFIX name, threadId, timeoutSec, TimeUnit.SECONDS);//避免拆箱导致空指针使用Boolean.TRUE.equals方法返回结果return Boolean.TRUE.equals(success);}Overridepublic void unlock() {// 获取线程标识String threadId ID_PREFIX Thread.currentThread().getId();// 获取锁标识String lockID stringRedisTemplate.opsForValue().get(KEY_PREFIX name);// 判断标识是否一致if(threadId.equals(lockID))stringRedisTemplate.delete(KEY_PREFIX name);} }分布式锁原子性问题  如上图所示线程1执行业务结束后进行释放锁的操作在对锁的标识进行判断后开始释放锁。但是线程1在判断结束到释放锁的期间受到了阻塞遇到JVM垃圾回收机制时会暂停程序导致阻塞这时线程2获取锁。当线程1恢复后继续进行释放锁的操作将会误删线程2的锁。我们前面设置了锁标识并且要求在释放锁之前需要做一个判断但在判断可以释放锁后如果遇到了阻塞将可能导致上图所示的误删操作。 解决方法我们需要实现判断和释放锁这两条命令的原子性问题。 Lua脚本 Redis提供了Lua脚本功能在一个脚本中编写多条Redis命令能够确保多条命令执行时的原子性。Lua是一种编程语言其基本语法可以参考网站Lua 教程 | 菜鸟教程。这里重点介绍Redis提供的调用函数我们可以使用lua去操作redis以保证多条redis命令的原子性这样就可以实现拿锁、判断、删锁多条命令的原子性动作了作为一名Java程序员这一块并不需要大家过于精通只需要知道它有什么作用即可。 编写脚本  我们需要在resources文件中新建.lua文件如果没有该新建项需要下载EmmyLua插件并在其中添加下图中的脚本内容。 代码优化 优化后的代码如下。 public class SimpleRedisLock implements ILock {private String name;private StringRedisTemplate stringRedisTemplate;public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {this.name name;this.stringRedisTemplate stringRedisTemplate;}private static final String KEY_PREFIX lock:;private static final String ID_PREFIX UUID.randomUUID().toString(true) -;private static final DefaultRedisScriptLong UNLOCK_SCRIPT;//初始化UNLOCK_SCRIPTstatic {UNLOCK_SCRIPT new DefaultRedisScript();UNLOCK_SCRIPT.setLocation(new ClassPathResource(unlock.lua));//初始化返回值UNLOCK_SCRIPT.setResultType(Long.class);}Overridepublic boolean tryLock(long timeoutSec) {// 获取线程标识String threadId ID_PREFIX Thread.currentThread().getId();// 获取锁并设置锁标识、添加时间Boolean success stringRedisTemplate.opsForValue().setIfAbsent(KEY_PREFIX name, threadId, timeoutSec, TimeUnit.SECONDS);//避免拆箱导致空指针使用Boolean.TRUE.equals方法返回结果return Boolean.TRUE.equals(success);}Overridepublic void unlock() {// 调用lua脚本stringRedisTemplate.execute(UNLOCK_SCRIPT,//要求传入KEYS集合使用Collections单元素集合工具Collections.singletonList(KEY_PREFIX name),//线程标识ID_PREFIX Thread.currentThread().getId());}/* Overridepublic void unlock() {// 获取线程标识String threadId ID_PREFIX Thread.currentThread().getId();// 获取锁标识String lockID stringRedisTemplate.opsForValue().get(KEY_PREFIX name);// 判断标识是否一致if(threadId.equals(lockID))stringRedisTemplate.delete(KEY_PREFIX name);}*/ } 总结  基于Redis的分布式锁实现思路 · 利用set nxex获取锁并设置过期时间保存线程标识 · 释放锁时先判断线程标识是否与锁标识一致若一致则删除锁 特性 · 利用set nx满足互斥性 · 利用set ex保证故障时锁依然能释放避免死锁提高安全性 · 利用redis集群保证高可用和高并发特性本文未涉及
http://www.w-s-a.com/news/278618/

相关文章:

  • 世界网站制作帮助做职业规划的网站
  • wordpress 查看站点ppt素材大全免费图片
  • 网站做弹幕广告有什么兼职做it的网站
  • 什么公司做网站出名广州做外贸网站公司
  • 源码网站取名企业网站怎么做百度
  • 织梦网站如何打通百度小程序深圳网站设计灵点网络品牌
  • 做网站网关备案展厅设计风格
  • 唐山网站建设费用网站title优化
  • 网站建设公司做销售好不好海南在线新闻中心
  • title 镇江网站建设wordpress 获取用户密码
  • 品牌型网站建设wordpress+js插件开发教程
  • 免费注册微信网站国家企业年审营业执照官网
  • 建设银行网站 无法访问东莞淘宝运营
  • 做家电网站做网站美工需要会什么软件
  • 深圳营销型定制网站开发1000建设银行网站特点分析
  • 安装网站系统重庆知名网站
  • 巴彦淖尔市 网站建设怀化北京网站建设
  • 内部网站管理办法建立网站后台
  • 自学考试网站建设与管理郑州网站建设开拓者
  • 宁夏制作网站公司慈溪建设集团网站
  • 国家企业官方网站查询系统站酷设计网站官网入口文字设计
  • 彩票网站开发制作需要什么wordpress连接微博专业版v4.1
  • 孝感建设银行官网站百度一下你就知道啦
  • 做网站如何做视频广告制作公司简介怎么写
  • 做网站 买空间商务网站内容建设包括
  • 萝岗网站建设为什么点不开网站
  • 惠州网站制作询问薇北京网站建设最便宜的公司
  • 注册网站英语怎么说wordpress 3.8.3
  • 甘肃张掖网站建设网站开发软件是什么专业
  • 海口省建设厅网站网站数据库怎么做同步