首页 > 数据库 >redis完成分布式锁

redis完成分布式锁

时间:2023-04-26 11:13:44浏览次数:40  
标签:redisson redis Redis 缓存 完成 import 分布式 stock

1. 正文

1. Redis完成分布式锁
2. redis的面试题。

2. 缓存

当执行增删改操纵时必须保证缓存和数据库数据一致性。---删除缓存

    @Override
    public Dept insert(Dept dept) {
        int i = deptMapper.insert(dept);
        return dept;
    }

    @Override
    public Integer delete(Integer id) {
        redisTemplate.delete("dept::"+id); //缓存与数据库数据保持一致
        int i = deptMapper.deleteById(id);
        return i;
    }

    @Override
    public Dept update(Dept dept) {
        //删除
        redisTemplate.delete("dept::"+dept.getDeptno()); //缓存与数据库数据保持一致
        int i = deptMapper.updateById(dept);
        return dept;
    }

3. redis使用分布式锁

(1)通过使用jmeter压测工具测试

同一个库存数被多个线程卖,线程安全问题。---思考:之间出现线程安全问题时如何解决。

可以使用锁解决:----synchronized和Lock锁

package com.lht.service;

import com.lht.mapper.StockMapper;
import com.lht.pojo.Stock;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import javax.annotation.Resource;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @author 卢海腾
 * @date 2023/4/25 17:06
 */
@Service
public class StockService_lock_syn {
    @Resource
    private StockMapper stockMapper;
    public static  Object o = new Object();
    Lock lock =  new ReentrantLock();

    public String jianStock(Integer pid){
        try{
            lock.lock();;
            Stock stock = stockMapper.selectById(pid);
            if (stock.getNum() > 0){
                stock.setNum(stock.getNum() - 1);
                stockMapper.updateById(stock);
                System.out.println("库存剩余数量:" + stock.getNum());
                return "减库存成功";
            }else {
                System.out.println("库存不足");
                return "减库存失败";
            }
        }
        finally {
            lock.unlock();
        }
    }
}

上面的synchronized或Lock锁是否适合集群模式|分布式系统。不适合、因为synchronized都是基于JVM的本地锁。

需要在idea中跑项目的集群

配置nginx

启动nginx

jmeter压测

两台集群出现重卖问题。

3.2 使用redis来解决分布式锁

  package com.lht.service;

import com.lht.mapper.StockMapper;
import com.lht.pojo.Stock;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.core.ValueOperations;
import org.springframework.stereotype.Service;

import java.util.concurrent.TimeUnit;

/**
 * @author 卢海腾
 * @date 2023/4/25 20:13
 */
@Service
public class StockService_lock_syn2 {

    @Autowired
    private StockMapper stockMapper;
    @Autowired
    private StringRedisTemplate redisTemplate;
    public String jianStock(Integer pid) {
        ValueOperations<String, String> forValue = redisTemplate.opsForValue();

        while (!forValue.setIfAbsent("product::" + pid, "", 30, TimeUnit.SECONDS)) {
            try {
                Thread.sleep(20);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        try {
            Stock stock = stockMapper.selectById(pid);
            if (stock.getNum() > 0) {
                stock.setNum(stock.getNum() - 1);
                stockMapper.updateById(stock);
                System.out.println("库存剩余数量:" + stock.getNum());
                return "库存减少成功";
            } else {
                System.out.println("库存不足");
                return "库存减少失败";
            }
        }finally {
            redisTemplate.delete("product::"+pid);//释放锁
        }
    }
}

如果你的业务代码的执行时间超过30s,当前线程删除的是其他线程的锁资源。 --watchDog机制

每个10s检测当前线程是否还持有所资源,如果持有则为当前线程延迟。---可以自己设置watchDog机制,---第三方Redission完美的解决分布式锁。

3.3 redisson完美解决redis超时问题

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

(2)main函数

@Bean //创建redisson交于spring容器来管理
    public RedissonClient redisson() {
        Config config = new Config();
        config.useSingleServer().setAddress("redis://192.168.223.166:6379");
        RedissonClient redisson = Redisson.create(config);
        return redisson;
    }

(3)使用

    package com.lht.service;

import com.lht.mapper.StockMapper;
import com.lht.pojo.Stock;
import org.redisson.Redisson;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;

import javax.annotation.Resource;
import java.util.concurrent.TimeUnit;

/**
 * @author 卢海腾
 * @date 2023/4/25 20:25
 */
public class StockService_lock_syn3 {
    @Resource
    private StockMapper stockMapper;

    @Resource
    private RedissonClient redisson;


        public  String jianStock(Integer pid){
            RLock lock = redisson.getLock("product::" + pid);
            try {
                lock.lock(30, TimeUnit.SECONDS);//加锁: 如果程序执行是出现一次
                //1. 查询指定的商品库存
                Stock stock = stockMapper.selectById(pid);
                if (stock.getNum() > 0) {
                    //2.库存减1
                    stock.setNum(stock.getNum() - 1);
                    stockMapper.updateById(stock);
                    System.out.println("库存剩余数量:" + stock.getNum());
                    return "减库存成功";
                } else {
                    System.out.println("库存不足");
                    return "库存减失败";
                }
            }finally {
                lock.unlock();
            }

        }

    }


4. redis面试题

一、Redis是什么?

redis是使用C语言编写的一个高速缓存数据,它以key-value形式存储数据,而且它支持的数据类型非常丰富。

二、Redis都有哪些使用场景?

  1. 热点数据的缓存
  2. 计时器
  3. 排行榜。
  4. 实现分布式锁
  5. 使用session的共享--后面项目时使用

四、Redis支持的数据类型有哪些?

  1. String 2. Hash 3.List 4.Set 5.ZSet

五、Redis为什么是单线程的?

因为Redis是基于内存的操作,CPU不是Redis的瓶颈,Redis的瓶颈最有可能是机器内存的大小或者网络带宽。既然单线程容易实现,而且CPU不会成为瓶颈,那就顺理成章地采用单线程的方案了

六、Redis真的是单线程的吗?

并不是真的单线程,比如:RDB---Bgsave时,创建一个子线程

七、Redis持久化有几种方式?

RDB(Redis Database):指定的时间间隔能对你的数据进行快照存储。

AOF:(Append Only File):每一个收到的写命令都通过 write 函数追加到文件中。

八、什么是缓存穿透?怎么解决?

  1. 查询的数据在数据库中不存在,缓存中也不存在,这时有可能有人恶意访问这种数据。这些请求都会访问数据库,从而出现数据库压力过大。

情景: 比如id不合法---

​ 确实数据库中不存在。

解决: 1. 在controller加校验

      2. 我们可以在缓存中存入一个空对象,但是对象的过期时间不要太长,一般不会超过5分钟。 
         3. 可以使用布隆过滤器。----自己查阅资料

九、怎么保证缓存和数据库数据的一致性?

  1. 设置合理的过期时间
  2. 当执行增删改时需要删除缓存数据

十、Redis,什么是缓存雪崩?怎么解决?

缓存雪崩:就是在某一时刻出现大量数据过期,而这时就有大量的请求访问该数据,这种现象叫做缓存雪崩。

什么情况下会出现大量数据过期:

  1. 项目刚刚上线
  2. redis服务器宕机
  3. 缓存数据真实过期

解决方案:

  1. 上线前预热数据。
  2. 集群
  3. 设置过期时间时要分散设置。

十一、Redis怎么实现分布式锁?

使用redis中的setnx命令--占锁,当业务代码执行完毕后是否锁资源,而释放锁的命令是del。

十二. redis在实现分布式锁式有什么缺陷

超时问题: 业务代码执行时间超过锁时间。使用:watchDog机制。 我们使用第三方:redisson

十三、Redis 淘汰策略有哪些?

标签:redisson,redis,Redis,缓存,完成,import,分布式,stock
From: https://www.cnblogs.com/gl0806/p/17353830.html

相关文章

  • Redis Server监控数据采集
    RedisServer监控数据采集ping,infoall,slowlogget/len/reset/clusterinfo/configgetRedis存活监控    redis-cli-h[ip]-p[port]-a[pwd]redis存活监控(redis_alive):redis本地监控agent使用ping,如果指定时间返回PONG表示存活,否则redis不能响应请求,......
  • 分布式编译系统的搭建
    GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源。GreatSQL是MySQL的国产分支版本,使用上与MySQL一致。作者:dan文章来源:GreatSQL社区原创由于MySQL源码编译单机耗费的时间过于长,最近MySQL变成8.0.27以后编译时间明显更耗时了,并且办公室内有多余的空闲......
  • Redis - -分布式锁
    Redis完成分布式锁#1.1缓存>当执行增删改操纵时必须保证缓存和数据库数据一致性。---删除缓存```java@OverridepublicDeptinsert(Deptdept){inti=deptMapper.insert(dept);returndept;}@OverridepublicIntegerd......
  • 这可能是最全面的Redis面试八股文了
    Redis连环40问,绝对够全!Redis是什么?Redis(RemoteDictionaryServer)是一个使用C语言编写的,高性能非关系型的键值对数据库。与传统数据库不同的是,Redis的数据是存在内存中的,所以读写速度非常快,被广泛应用于缓存方向。Redis可以将数据写入磁盘中,保证了数据的安全不丢失,而且Redis......
  • 滑动窗口算法实现分布式第三方请求限频
    一.业务背景 第三方服务接口存在频率调用限制(例如,1s5次,超过5次返回超出频率),己方服务存在并发处理的情况,为了保证服务的成功率,且达到第三方限制的最大吞吐量,故需要一个限频调用的算法二.实现思路常见限频算法一般有五种,漏桶算法、令牌桶算法、固定窗口算法,滑动窗口算法,漏斗算......
  • 美国销售税合规平台TaxCloud完成2000万美元融资
    猛兽财经获悉,总部位于加州诺沃克面向电子商务企业的销售税合规平台TaxCloud近期宣布完成2000万美元融资。本轮融资由CamberPartners领投。该公司打算利用这笔资金继续为其客户提供服务,同时扩大其产品供应、营销和销售。作为投资的一部分,TaxCloud宣布任命NateGilmore为首席执行......
  • “源擎”云原生分布式核心业务系统有什么产品优势?
     “源擎”核心系统利用云原生、分布式、微服务技术,基于企业架构设计思想,构建了基础服务、业务服务、交易中心以及系列支撑组件,包含业务架构和多个微服务应用。业务架构中,交易中心为银行提供了更灵活的选择,支持产品粒度的功能可替换,同时也能快速引入新的产品服务,支持未来业务发展,实......
  • 下一代大数据分布式存储技术Apache Ozone初步研究
    @目录概述定义特性架构总体架构写数据读数据部署安装方式安装Docker启动Docker-compose启动企业预置型(OnPremise)安装实践命令行接口Ofs(Hadoop兼容)ReconAPI概述定义ApacheOzone官网地址https://ozone.apache.org/最新版本1.3.0ApacheOzone官网最新文档地址http......
  • redis题目(二)
    1.Redis提供了多种AOF缓冲区同步文件策略,由参数appendfsync控制。参数appendfsync的可选值不包括以下哪个选项()alwayseverymineverysecno2.下面关于Redis中内存划分的说法,错误的是()Redis在存储对象时会将对象进......
  • OpenSUSE Leap 15.3 系统分析与调整指南【翻译完成】
    在线阅读在线阅读(Gitee)ApacheCN学习资源贡献指南本项目需要校对,欢迎大家提交PullRequest。请您勇敢地去翻译和改进翻译。虽然我们追求卓越,但我们并不要求您做到十全十美,因此请不要担心因为翻译上犯错——在大部分情况下,我们的服务器已经记录所有的翻译,因此您不必担心会因为您的......