Redssion+Lua实现原子减库存防止超卖

在秒杀环节中超卖一直是个大忌,我们允许一定程度上的少卖但是我们绝不允许出现超卖的现象.设想一下,那种一元抢购千元手机的秒杀活动,如果超卖了,损失肯定是比较大的.前段时间楼主做过一个类似于秒杀的项目,所以做了一下总结,当然实际项目要比我下面讲的要复杂.

1. 说明

  Lua 是一种轻量小巧的脚本语言,采用C语言实现,一方面减小了访问redis的网络开销,另一方面天然的原子性有效的解决了事务的难题,但是redis集群对lua集群的支持有限的,集群有一个slot的概念,Redis要求单个Lua脚本操作的key必须在同一个节点上,但是Cluster会将数据自动分布到不同的节点(虚拟的16384个slot,具体看官方文档),但是集群模式下不同的key根据redis的hash算法会分布在不同的槽,集群是不支持多key查询的,解决办法是使用redis的hash tag,存入的时候把要多key查询的 key用{}括起来。keySlot算法中,如果key包含{},就会使用第一个{}内部的字符串作为hash key,这样就可以保证拥有同样{}内部字符串的key就会拥有相同slot。也可以使用代理来完成Lua在集群模式下的实现.本文redis是非集群模式.

2. 背景

  现在我要实现这样一个功能,我们只有两件商品,商品hh初始化库存为7,商品newday初始化库存为6,现在用户每次都捆绑买这两件商品,只要有一个商品库存不足,本次活动结束.

3. 基于Redssion+Lua实现

  redission实现了JDK中的Lock接口,所以使用方式上和我们用lock一样,只是Redssion的锁是分布式的,这也是redis官方现在比较力推的.

下面分享一下干货,毕竟talk is cheap.

逻辑控制层代码:

package com.example.demo.controller;

import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;

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


@RestController
public class RedissionTest {

    @Resource
    private RedissonClient redissonClient;

    @Resource
    private DefaultRedisScript<Boolean> redisScript;

    @Resource
    private StringRedisTemplate stringRedisTemplate;

    @RequestMapping(value = "test", method = {RequestMethod.GET})
    public void test() {
        String ss="66666";
        System.out.println(ss);

        RLock lock = redissonClient.getLock(ss);
        try {
            // 1. 最常见的使用方法
            //lock.lock();
            // 2. 支持过期解锁功能,10秒钟以后自动解锁, 无需调用unlock方法手动解锁
            //lock.lock(10, TimeUnit.SECONDS);
            // 3. 尝试加锁,最多等待2秒,上锁以后8秒自动解锁
            boolean res = lock.tryLock(2, 8, TimeUnit.SECONDS);
            if (res) { //成功
                //处理业务
                System.out.println("33333333{} " + ss);
                List<String> keys = Arrays.asList("hh","newday");

                Boolean execute = stringRedisTemplate.execute(redisScript, keys);


                System.out.println("999");
                System.out.println("**** {}" + execute);

            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            //释放锁
            System.out.println("8888");
            lock.unlock();
        }

    }
}

Redssion配置(本文以单机模式演示):

package com.example.demo.config;

import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.client.codec.StringCodec;
import org.redisson.config.Config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;


@Configuration
public class RedissonConfig {
    @Bean
    public RedissonClient getRedisson(){
        Config config = new Config();
        //单机模式  依次设置redis地址和密码
        config.useSingleServer().
                setAddress("redis://127.0.0.1:6379?auth=123456")
                .setPassword("123456");
        config.setCodec(new StringCodec());
        return Redisson.create(config);
    }
}

初始化Lua配置:

package com.example.demo.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.ClassPathResource;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.scripting.support.ResourceScriptSource;

@Configuration
public class LuaConfiguration {
    @Bean
    public DefaultRedisScript<Boolean> redisScript() {
        DefaultRedisScript<Boolean> redisScript = new DefaultRedisScript<>();
        redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource("Test.lua")));
        redisScript.setResultType(Boolean.class);
        return redisScript;
    }
}

3. 压力测试

这里我采用的压力测试工具为siege,在mac上面使用起来还是比较简单的.
为了有效的模拟实际的效果,并发50个线程循环5秒在线压力测试,5秒内的请求数远超过商品库存数.

siege.png

3.1测试结果

从执行结果来看,我们有效的控制了超卖的现象.

Redis执行结果变化:

1.jpg

控制台执行过程:

66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
33333333{} 66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
999
**** {}true
8888
66666
33333333{} 66666
66666
999
**** {}true
8888
33333333{} 66666
999
**** {}true
8888
66666
33333333{} 66666
66666
999
**** {}true
8888
66666
33333333{} 66666
999
**** {}true
8888
66666
33333333{} 66666
999
**** {}true
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888

66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
3333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
66666
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false

**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
66666
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
66666
66666
66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false

33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
66666

33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false

33333333{} 66666
999
**** {}false
8888
33333333{} 66666
66666
999
**** {}false
8888
33333333{} 66666
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
33333333{} 66666
999
**** {}false
8888
66666
66666
33333333{} 66666
999
**** {}false
8888
33333333{} 66666
999
**** {}false

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容