synchronized锁和redis分布式锁的使用

准备工作

1.商品库存都以50为例存在redis中

image

2.商品购买接口

Controller

@RestController
@RequestMapping("/test")
public class TestController {

    @Autowired
    private RedisTemplate<String, Object> redisTemplate;

    @GetMapping("/buy")
    public String buy(){
        //查询当前商品库存
        Integer store = (Integer) redisTemplate.opsForValue().get("store");

        if(store > 0){
            //库存不为0则库存减一
            store =  store - 1;
            redisTemplate.opsForValue().set("store",store);
            System.out.println("购买成功,购买后库存为:"+ store);
        }else{
            System.out.println("购买失败");
        }
        return "ok";
    }
}

3.使用jMeter进行压测

设置访问地址:http://localhost/lock-service/test/buy

image

设置并发数,这里是0s内发送200个请求。
image

测试

1.未加锁

image

image

结果:原库存为50,但成功卖出了200件商品,卖出200件商品后库存变为20,商品超卖了。

2.加synchronized锁

重点为锁对象的选取,这里使用商品id字符串,对应的常量池中的引用做为锁对象。

学习synchronized的时候一般都是用this做为锁对象的,这里如果使用this,购买不同商品时,也会争夺同一把锁,效率较低。

@GetMapping("/buy")
public String buy(){
    //使用id的字符串做为锁对象,intern方法 (返回常量池中该字符串的引用)
    String poductId = "100";
    String lock = poductId.intern();
    synchronized (lock.getClass()){
        //查询当前商品库存
        Integer store = (Integer) redisTemplate.opsForValue().get("store");
        if(store > 0){
            //库存不为0则库存减一
            store =  store - 1;
            redisTemplate.opsForValue().set("store",store);
            System.out.println("购买成功,购买后库存为:"+ store);
        }else{
            System.out.println("购买失败");
        }
    }
    return "ok";
}
image

image

结果:原库存为50,成功卖出了50件商品,卖出50件商品后库存变为0,加锁成功

3.加synchronized锁,并启动多个商品服务

image

我这里启动两个商品购买服务,锁还是使用synchronized锁,使用SpringClouldGateway负载均衡调用商品服务。

商品服务实例1


image

商品服务实例2


image

结果:通过实例1和实例2的结果可以看出,至少36、35、22、18存在着重复卖出,商品超卖了。因为两个商品服务是启动在两个jvm中,synchronized无法实现跨虚拟机加锁,所以分布式系统中不能使用synchronized锁。

4.加Redis分布式锁,并启动多个商品服务

官方文档

a.引入依赖,SpringBoot版本自选

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

<dependency>
    <groupId>org.springframework.integration</groupId>
    <artifactId>spring-integration-redis</artifactId>
    <version>5.2.0.RELEASE</version>
</dependency>

b.yml配置

spring: 
  redis:
    database: 0 # Redis数据库索引(默认为0)
    host: 127.0.0.1
    port: 6379
    jedis:
      pool:
        max-active: 60 # 连接池最大连接数(使用负值表示没有限制)
        max-idle: 60 # 连接池中的最大空闲连接
        min-idle: 20 # 连接池中的最小空闲连接

c.注入Bean对象

@Bean
RedisLockRegistry redisLockRegistry(RedisConnectionFactory connectionFactory) {
    return new RedisLockRegistry(connectionFactory, "redisLock", 5000L);
}

注 RedisLockRegistry 提供两个构造方法, 上述示例, 最后一个参数为 默认过期时间

To avoid “hung” locks (when a server fails), the locks in this registry are expired after a default 60 seconds, but you can configure this value on the registry. Locks are normally held for a much smaller time.

d.使用Redis锁

@Autowired
private RedisLockRegistry redisLockRegistry;

@GetMapping("/buy")
public String buy(){
    String poductId = "100";
    Lock lock = redisLockRegistry.obtain("buyRedis:" + poductId);
    try{
        lock.lock();
        //查询当前商品库存
        Integer store = (Integer) redisTemplate.opsForValue().get("store");

        if(store > 0){
            //库存不为0则库存减一
            store =  store - 1;
            redisTemplate.opsForValue().set("store",store);
            System.out.println("购买成功,购买后库存为:"+ store);
        }else{
            System.out.println("购买失败");
        }
    }finally {
        lock.unlock();
    }

    return "ok";
}

e.启动两个商品服务进行测试

因为我设置50个商品时,总是在某一个实例里卖完了所有商品(执行太快了),所以我把商品数量设成了200.
实例1


image

实例2


image

image.png

结果:原库存为200,成功卖出了200件商品,卖出200件商品后库存变为0,加分布式锁成功。

自己实现一个redis锁

1.代码

@RestController
@RequestMapping("/test")
public class TestController {

    @Autowired
    private RedisTemplate<String, Object> redisTemplate;

    @GetMapping("/buy")
    public String buy(){
        String lockKey = "productKey_100";

    //这个锁是阻塞型的锁且会自动增加锁的生存时间
        lock(lockKey);

        try {
        //设置购买操作的时间大于锁的默认生存时间9s,测试锁的续命效果
            Thread.sleep(12000L);
            buyProduct();
        }catch (Exception e){
            System.out.println("休眠错误");
        }finally {
            unlock(lockKey);
        }

        return "ok";
    }

    /**
     * 购买商品
     */
    private void buyProduct(){
        //查询当前商品库存
        Integer store = (Integer) redisTemplate.opsForValue().get("store");

        if(store > 0){
            //库存不为0则库存减一
            store =  store - 1;
            redisTemplate.opsForValue().set("store",store);
            System.out.println("购买成功,购买后库存为:"+ store);
        }else{
            System.out.println("购买失败");
        }
    }

    /**
     * 加锁
     */
    private void lock(String lockKey){
        String uuid = UUID.randomUUID().toString();
        while(true){
            if(redisTemplate.opsForValue().setIfAbsent(lockKey,uuid , 9, TimeUnit.SECONDS)){
                Thread thread = new Thread(new Runnable() {
                    @Override
                    public void run() {
                        Timer timer = new Timer();

                        TimerTask timerTask = new TimerTask() {
                            @Override
                            public void run() {
                //锁存在则将生存时间重置为9s
                                String o =(String) redisTemplate.opsForValue().get(lockKey);
                                if(uuid.equals(o)){
                                    redisTemplate.expire(lockKey,9,TimeUnit.SECONDS);
                                }else{
                                    timer.cancel();
                                }
                            }
                        };
            //定时器启动3s后执行第一次,之后每隔3s执行一次
                        timer.schedule(timerTask,3000L,3000L);
                    }
                });

                thread.run();
                break;
            }
        }
    }

    /**
     * 解锁锁
     */
    private void unlock(String lockKey){
        redisTemplate.delete(lockKey);
    }
}

2.效果

设置30个库存,并发发送30个购买请求,执行了12*30秒,得到结果如下


image

image

image

30个商品都成功卖出,卖出后库存为0,加锁成功,锁的阻塞效果和锁的续命效果也成功了。

容易踩的坑

1.事务中加锁,会导致锁失效

原因:由于事务是加锁前开启的,锁是事务未提交前释放的,此时其他线程拿到锁之后进行锁住的代码块,读取的库存数据不是最新的。

解决方法:我们可以在@Transactional注释的方法之前就加上锁,在还没有开事务之前就加锁,那么就可以保证线程的安全性,从而不会出现脏读和数据不一致性等情况。

2.锁对象的选取

锁对象的选取要和实际业务关联,例如商品购买,就要和商品关联。一律使用当前类(this)做锁对象效率不高。

同时对于同一商品的锁对象要是相同对象(同一引用的对象)。

3.单实例服务可以使用非分布式的锁,多实例服务只能使用分布式锁

分布式锁的实现方案有很多

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,383评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,522评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,852评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,621评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,741评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,929评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,076评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,803评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,265评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,582评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,716评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,395评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,039评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,027评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,488评论 2 361
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,612评论 2 350