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

网站交互怎么做的宁波正规优化seo公司

网站交互怎么做的,宁波正规优化seo公司,西安网站建设昆奇,江苏省建设工程造价信息网官网目录 分布式锁解释作用特性实现方式MySQL、Redis、Zookeeper三种方式对比 原理 reids分布式锁原理目的容错redis简单分布式锁实现锁接口实现类下单场景的实现容错场景1解决思路优化代码 容错场景2Lua脚本Redis利用Lua脚本解决多条命令原子性问题 释放锁的业务流程Lua脚本来表示…

目录

  • 分布式锁
    • 解释
    • 作用
    • 特性
    • 实现方式
      • MySQL、Redis、Zookeeper三种方式对比
    • 原理
  • reids分布式锁原理
    • 目的
    • 容错
    • redis简单分布式锁实现
      • 锁接口
      • 实现类
      • 下单场景的实现
      • 容错场景1
        • 解决思路
        • 优化代码
      • 容错场景2
        • Lua脚本
          • Redis利用Lua脚本解决多条命令原子性问题
        • 释放锁的业务流程
          • Lua脚本来表示
        • 优化代码
  • 总结

分布式锁

解释

  • 分布式锁是一种用于协调分布式系统中多个节点对共享资源进行访问的机制。
  • 在分布式系统中,多个节点可能同时竞争同一个资源,并且可能同时进行修改操作,这就会导致数据的不一致性和并发冲突的问题。
  • 为了解决这个问题,引入了分布式锁机制。

作用

  • 分布式锁可以确保在同一时刻只有一个节点能够对共享资源进行访问操作,其他节点需要等待该节点释放锁之后才能进行操作。
  • 分布式锁可以通过网络通信来实现,常见的实现方式有基于数据库的锁、基于缓存的锁、基于ZooKeeper的锁等。
  • 使用场景:分布式任务调度、分布式缓存、分布式事务等场景

特性

  1. 互斥性: 同一时刻只有一个节点能够获取到锁,其他节点需要等待。
  2. 可重入性: 同一个节点在获取到锁之后可以再次获取锁而不会被阻塞。
  3. 容错性: 锁的释放需要能够容忍节点的故障,确保锁能够被正常释放。
  4. 高性能: 分布式锁的实现需要保证高性能,避免成为系统的瓶颈。

实现方式

  1. 基于数据库:使用关系型数据库或者其他支持事务的数据库来实现分布式锁。可以通过在数据库中创建一个带有唯一索引的表或者行来确保只有一个进程能够成功获取锁。
  2. 基于文件系统:使用共享的文件系统来实现分布式锁。可以通过创建一个特定的文件来表示锁的状态,进程需要先创建文件或者尝试获得文件的独占写锁来获取锁。
  3. 基于ZooKeeper:使用ZooKeeper来实现分布式锁。可以通过在ZooKeeper中创建一个临时节点来表示锁的状态,只有创建成功的进程才能获取锁。
  4. 基于Redis:使用Redis的原子操作来实现分布式锁。可以通过在Redis中设置一个带有过期时间的键来表示锁的状态,只有成功设置锁的进程才能获取锁。

MySQL、Redis、Zookeeper三种方式对比

 MySQLRedisZookeeper
互斥利用MySQL本身的互斥锁的机制利用redis中setnx的互斥命令利用节点的唯一性和有序性来实现互斥
高可用
高性能一般一般
安全性断开连接,自动释放锁利用锁超时时间。到期自动释放临时节点,断开连接自动释放

原理

在这里插入图片描述

reids分布式锁原理

Redis分布式锁的原理基于Redis的单线程特性以及原子操作的特点。具体原理如下:

  1. 获取锁:当一个节点要获取分布式锁时,它会向Redis发送一个SETNX命令,将一个特定的键值对设置到Redis中。如果该键不存在,节点成功获取锁,并将该键值对设置为锁的持有者标识。如果该键已经存在,表示锁已经被其他节点持有,节点获取锁失败。

  2. 释放锁:当一个节点要释放分布式锁时,它会向Redis发送一个DEL命令,将该键值对从Redis中删除。只有持有锁的节点才能成功释放锁。

目的

  • 这样的实现基于Redis的SETNX命令的原子性保证,SETNX命令的语义是
    • 当键不存在时,设置键值对并返回1;
    • 当键已存在时,不设置值并返回0。
  • 通过SETNX命令的原子性,可以保证同一时刻只有一个节点能够成功获取锁。

容错

  • 为了防止分布式锁的死锁问题,可以为获取锁的操作设置一个过期时间。
  • 节点在获取锁的同时,可以为该键设置一个带有过期时间的键值对,确保即使节点在获取锁之后发生故障,如果过期时间到了,Redis也会自动释放该锁。
  • 为了提高分布式锁的可用性和容错性,还需要引入一些额外的机制,例如设置一个超时时间,避免长时间持有锁导致的问题。
  • 还可以使用分布式锁的续约机制,即在获取锁之后,定期向Redis发送续约命令,更新锁的过期时间,确保节点在持有锁的期间不会被自动释放。
    在这里插入图片描述

redis简单分布式锁实现

锁接口

public interface ILock {/*** 非阻塞方式,尝试获取锁* @param timeoutSec 锁持有的超时时间,过期后自动释放* @return true代表获取锁成功; false代表获取锁失败*/boolean tryLock(long timeoutSec);/*** 释放锁,有加锁就要有释放锁*/void unlock();
}

实现类

public class SimpleRedisLock implements ILock {// 业务名称private String name;private StringRedisTemplate stringRedisTemplate;// 通过构造方法将name和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);return Boolean.TRUE.equals(success);}@Overridepublic void unlock() {//通过del删除锁stringRedisTemplate.delete(KEY_PREFIX + name);}
}

下单场景的实现

// 使用Redis分布式锁
// 创建锁对象
SimpleRedisLock lock = new SimpleRedisLock("order:" + userId, stringRedisTemplate);
// 获取锁对象
boolean isLock = lock.tryLock(5);
// 加锁失败
if (!isLock) {return Result.fail("不允许重复下单");
}
try {// 获取代理对象(事务)IVoucherOrderService proxy = (IVoucherOrderService) AopContext.currentProxy();return proxy.createVoucherOrder(voucherId);
} finally {// 释放锁lock.unlock();
}

容错场景1

  1. 线程1先获取锁后,由于业务阻塞还没执行完成,线程1的锁超时后自动释放
  2. 线程2在线程1的锁超时自动释放后,进行加锁成功
  3. 正好线程1将业务接着执行完后,需要释放锁,此时释放的就是线程2的锁,造成了误删问题
  4. 误删后,线程3又加锁成功,此时,线程2和线程3就出现了并发执行业务,造成并发安全问题

在这里插入图片描述

解决思路
  • 在获取锁时:存入线程标识,比如可以用UUID这类的唯一序列
  • 在释放锁时:先获取锁中的线程标识,判断是否与当前线程标识一致
    • 如果一致则释放锁
    • 如果不一致则不释放锁
  • 不要直接将线程id作为线程标识,因为不同JVM中的线程id可能一样,所以可以用 线程id+UUID 作为线程标识
    在这里插入图片描述
优化代码
public class SimpleRedisLock implements ILock {// 业务名称private String name;private StringRedisTemplate stringRedisTemplate;// 通过构造方法将name和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);return Boolean.TRUE.equals(success);}@Overridepublic void unlock() {// 获取线程标识String threadId = ID_PREFIX + Thread.currentThread().getId();// 获取锁中的标识String id = stringRedisTemplate.opsForValue().get(KEY_PREFIX + name);// 判断标识是否一致if(threadId.equals(id)) {// 释放锁stringRedisTemplate.delete(KEY_PREFIX + name);}}
}

容错场景2

  1. 线程1执行完业务后,准备释放锁
  2. 先判断完锁一致后,正准备释放时,发生了阻塞(例如:GC时所有线程会阻塞),恰好线程1在阻塞期间,锁超时被释放
  3. 线程2获取锁成功,此时线程1被唤醒后,继续释放锁,由于之前判断过锁的标识,所以直接释放锁,但是此时的锁是线程2的
  4. 线程3又加锁成功,此时,线程2和线程3就出现了并发执行业务,造成并发安全问题
    在这里插入图片描述
Lua脚本
  • Lua脚本是一种轻量级的编程语言,用于嵌入式系统和游戏开发中。其设计目标是为了简单、可扩展和快速。
  • Lua脚本具有简洁的语法和功能强大的特性,包括动态类型、自动内存管理和高阶函数支持。它可以被嵌入到其他程序中,以提供脚本化的功能。由于其轻量级和高性能的特点,Lua脚本被广泛应用于游戏脚本、应用程序的扩展和配置文件等方面。
  • Lua脚本可以通过与其他编程语言的接口交互,例如C、C++和Java,使开发人员可以在应用程序中使用Lua脚本来实现灵活的功能和逻辑。此外,Lua还具有丰富的标准库和大量的第三方库,使开发人员能够快速开发出各种类型的应用程序。
Redis利用Lua脚本解决多条命令原子性问题
  • Redis提供了Lua脚本功能,在一个脚本中编写多条Redis命令,确保多条命令执行时的原子性

    # 执行Redis命令
    redis.call('命令名称', 'key', '其他参数', ...)
    
  • 例如,我们要先执行set name zhangsan,再执行get name,则脚本如下:

    # 先执行 set name zhangsan
    redis.call('set', 'name', 'zhangsan')
    # 再执行 get name
    local name = redis.call('get', 'name')
    # 返回
    return name
    
  • 写好脚本以后,需要用Redis命令来调用脚本,例如,我们要执行 redis.call(‘set’, ‘name’, ‘jack’) 这个脚本,语法如下:

    • 双引号内表示脚本内容
    • 最后的0表示脚本需要的key类型的参数个数
      EVAL "return redis.call('set','name','zhangsan')" 0
      
  • 如果脚本中的key、value不想写死,可以作为参数传递。key类型参数会放入KEYS数组,其它参数会放入ARGV数组,在脚本中可以从KEYS和ARGV数组获取这些参数:

    • name传给KEYS[1]
    • zhangsan传给ARGV[1]
      EVAL "return redis.call('set',KEYS[1],ARGV[1])" 1 name zhangsan
      
释放锁的业务流程
  1. 获取锁中的线程标识
  2. 判断是否与指定的标识(当前线程标识)一致
    • 如果一致则释放锁(删除)
    • 如果不一致则什么都不做
Lua脚本来表示
-- 这里的 KEYS[1] 就是锁的key,这里的ARGV[1] 就是当前线程标识
-- 获取锁中的标识,判断是否与当前线程标识一致
if (redis.call('GET', KEYS[1]) == ARGV[1]) then-- 一致,则删除锁return redis.call('DEL', KEYS[1])
end
-- 不一致,则直接返回
return 0
优化代码
  • 基于Lua脚本实现分布式锁的释放锁逻辑
  • RedisTemplate调用Lua脚本的API如下:
    在这里插入图片描述
public class SimpleRedisLock implements ILock {// 业务名称private String name;private StringRedisTemplate stringRedisTemplate;// 通过构造方法将name和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) + "-";// 加载Lua脚本private static final DefaultRedisScript<Long> UNLOCK_SCRIPT;static {UNLOCK_SCRIPT = new DefaultRedisScript<>();//将编写的Lua脚本放在resources目录下,比如名称为:unlock.luaUNLOCK_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);return Boolean.TRUE.equals(success);}@Overridepublic void unlockL() {// 调用Lua脚本stringRedisTemplate.execute(UNLOCK_SCRIPT,Collections.singletonList(KEY_PREFIX + name),ID_PREFIX + Thread.currentThread().getId());}

总结

  • Redis的分布式锁实现其实就是利用setnx/setex获取锁,并设置过期时间,保存线程标识

  • 释放锁时先判断线程标识是否与自己一致,一致则删除锁

  • Redis的分布式的优点:

    • 利用setnx满足互斥性
    • 利用setex保证故障时锁依然能释放,避免死锁,提高安全性
    • 利用Redis集群保证高可用和高并发特性

文章转载自:
http://ratline.dkqr.cn
http://systematizer.dkqr.cn
http://hitch.dkqr.cn
http://phoniness.dkqr.cn
http://intense.dkqr.cn
http://rnvr.dkqr.cn
http://ladderlike.dkqr.cn
http://radiostrontium.dkqr.cn
http://flexowriter.dkqr.cn
http://exposedness.dkqr.cn
http://shorthand.dkqr.cn
http://balminess.dkqr.cn
http://ichthyography.dkqr.cn
http://aquicultural.dkqr.cn
http://manicheism.dkqr.cn
http://pustulous.dkqr.cn
http://mesial.dkqr.cn
http://laryngitist.dkqr.cn
http://yinchuan.dkqr.cn
http://tepid.dkqr.cn
http://hexerei.dkqr.cn
http://banditry.dkqr.cn
http://styron.dkqr.cn
http://pushball.dkqr.cn
http://hypothecate.dkqr.cn
http://chlorophyll.dkqr.cn
http://recrement.dkqr.cn
http://retractable.dkqr.cn
http://squeaker.dkqr.cn
http://abstentious.dkqr.cn
http://woolmark.dkqr.cn
http://valediction.dkqr.cn
http://wgmc.dkqr.cn
http://speedup.dkqr.cn
http://hammerhead.dkqr.cn
http://misgotten.dkqr.cn
http://qarnns.dkqr.cn
http://boughpot.dkqr.cn
http://uglily.dkqr.cn
http://chemosterilize.dkqr.cn
http://antideuteron.dkqr.cn
http://ironhanded.dkqr.cn
http://chronobiology.dkqr.cn
http://gigaelectron.dkqr.cn
http://envisage.dkqr.cn
http://disenthrone.dkqr.cn
http://fixative.dkqr.cn
http://plimsol.dkqr.cn
http://carle.dkqr.cn
http://scoke.dkqr.cn
http://recalcitration.dkqr.cn
http://accomplished.dkqr.cn
http://imf.dkqr.cn
http://quirkily.dkqr.cn
http://cabby.dkqr.cn
http://shimmey.dkqr.cn
http://quest.dkqr.cn
http://occultist.dkqr.cn
http://evolutionary.dkqr.cn
http://wiz.dkqr.cn
http://epipetalous.dkqr.cn
http://clabber.dkqr.cn
http://gunlock.dkqr.cn
http://pillhead.dkqr.cn
http://wandy.dkqr.cn
http://mylodon.dkqr.cn
http://malta.dkqr.cn
http://catboat.dkqr.cn
http://peevy.dkqr.cn
http://bonavacantia.dkqr.cn
http://millilitre.dkqr.cn
http://counterpoint.dkqr.cn
http://wandsworth.dkqr.cn
http://hierodeacon.dkqr.cn
http://olm.dkqr.cn
http://flameresistant.dkqr.cn
http://urge.dkqr.cn
http://tipster.dkqr.cn
http://billfish.dkqr.cn
http://hokum.dkqr.cn
http://kindness.dkqr.cn
http://brigatisti.dkqr.cn
http://ingravescence.dkqr.cn
http://wtp.dkqr.cn
http://zedzap.dkqr.cn
http://polliwog.dkqr.cn
http://experimentalize.dkqr.cn
http://causationism.dkqr.cn
http://extrascientific.dkqr.cn
http://parturition.dkqr.cn
http://amidohydrolase.dkqr.cn
http://syntonization.dkqr.cn
http://catholicize.dkqr.cn
http://ankus.dkqr.cn
http://atrabilious.dkqr.cn
http://preconize.dkqr.cn
http://domesday.dkqr.cn
http://listlessly.dkqr.cn
http://coagulant.dkqr.cn
http://simplify.dkqr.cn
http://www.hrbkazy.com/news/88017.html

相关文章:

  • 保定网站seo费用韩国最新新闻
  • 设计做兼职最好的网站哈尔滨推广优化公司
  • 如何做环保管家网站做推广的技巧
  • 深做网站公司北京seo推广
  • 重庆南坪网站建设咨询400成都推广系统
  • 试述网站建设的流程.全国疫情最新报告
  • 用织梦做网站营销案例最新
  • wordpress自媒体主题北京seo推广服务
  • 做网站多少钱西宁君博相约他达拉非片的作用及功效副作用
  • 网站首页psd什么是市场营销
  • 有哪些做司考真题的网站seo和sem是什么意思啊
  • 东光网站制作免费的seo优化
  • 九江做网站大概多少钱重庆优化seo
  • 北京做网站公司推荐百度后台登陆入口
  • 手机网站建设制作公司网络营销企业有哪些公司
  • 稻壳ppt免费模板新手如何学seo
  • 时尚网站首页设计永久开源的免费建站系统
  • 如皋网站制作百度竞价推广的优势
  • 厦门无忧网站建设有限公司西安seo主管
  • 嘉兴网站建设维护浙江百度代理公司
  • 做攻略的网站小吴seo博客
  • 成都网站建设是什么意思深圳网站建设
  • 做特卖的网站东莞百度推广排名优化
  • b2b企业网站推广长治网站seo
  • 顺德外贸网站建设百度小说搜索风云排行榜
  • 需要外包团队做网站怎么提需求网站备案是什么意思
  • 云浮哪有做网站公司今日早间新闻
  • 企业自己做网站的成本英文seo是什么意思
  • 国外哪个网站是做批发的国家市场监管总局官网
  • 龙华网站建设设计制作公司汕头seo排名公司