黑马点评--优惠卷秒杀

news2025/2/25 20:58:04

黑马点评–优惠卷秒杀

全局ID生成器:

是一种在分布式系统下用来生成全局唯一ID的工具,一般要满足下列特性:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lvc2qD5J-1668574944843)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221113221848596.png)]

为了增加ID的安全性,我们可以不直接使用Redis自增的数值,而是拼接一些其它信息:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FM7N8l3N-1668574944844)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221113222812554.png)]

Redis自增ID策略:

  • 每天一个key,方便统计订单量
  • iD结构是时间戳+计数器
    /**
     * 开始时间戳
     */
    private static final long BEGIN_TIMESTAMP = 1640995200;
    /**
     * 序列号的位数
     */
    private static final int COUNT_BITS = 32;
    /**
     * id生成策略
     *
     * @param keyPrefix 业务前缀
     * @return
     */
    @Resource
    private StringRedisTemplate stringRedisTemplate;

    public long nextId(String keyPrefix) {
        //1.生成时间戳
        LocalDateTime now = LocalDateTime.now();
        long nowSecond = now.toEpochSecond(ZoneOffset.UTC);
        long timestamp = nowSecond - BEGIN_TIMESTAMP;
        //2.生成序列号
        //2.1获取当前日期,精确到天
        String date = now.format(DateTimeFormatter.ofPattern("yyyy:MM:dd"));
        //2.2自增长
        Long count = stringRedisTemplate.opsForValue().increment("icr:" + keyPrefix + ":" + date);
        //3.拼接并返回

        return timestamp << COUNT_BITS | count;
    }


    public static void main(String[] args) {

        LocalDateTime time = LocalDateTime.of(2022, 1, 1, 0, 0, 0);
        long second = time.toEpochSecond(ZoneOffset.UTC);
        System.out.println(second);
    }

测试ID自增生成策略:

500个线程每一个线程生成100id一共50000个id花的时间:

private ExecutorService service = Executors.newFixedThreadPool(500);
    @Test
    void testIdWorker() throws InterruptedException {
        CountDownLatch latch =new CountDownLatch(500);
        Runnable task = ()->{
            for (int i=0;i<100;i++){
                long id= redisIdWorker.nextId("order");
                System.out.println("id=" +id);
            }
            latch.countDown();
        };
        long start=System.currentTimeMillis();
        for (int i=0;i<500;i++){
            service.submit(task);
        }
        latch.await();
        long end=System.currentTimeMillis();
        System.out.println("end=" +(end-start));
    }

每个店铺都可以发布优惠卷:

当用户抢购时,就会生成订单并保存到tb_voucher_order这张表中,而订单表如果使用数据库自增ID就会存在一些问题:

  • id的规律太明显
  • 受单表数据量的限制

全局唯一ID生成策略:

  • UUID
  • Redis自增
  • sonwflake算法
  • 数据库自增

实现优惠券秒杀下单

每个店铺都可以发布优惠券,分为平价卷和特价卷。平价劵可以任意购买,而特价劵需要秒杀抢购:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lLOQ08x3-1668574944845)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114160638726.png)]

表关系如下:

  • tb_voucher:优惠劵的基本信息,优惠金额,使用规则等
  • tb_seckill_voucher:优惠劵的库存,开始抢购时间,结束抢购时间。特价优惠券才需要填写这些信息

在VoucherController中提供一个接口,可以添加秒杀优惠券:http://localhost:8081/voucher/seckill

{
"shopId":1,
"title":"100元代金券",
"subTitle":"周一至周五均可使用",
"rules": "全场通用\\n无需预约\\n可无限叠加\\不兑换、不找零\\n仅限堂食",
"payValue": 8000,
"actualValue": 10000,
"type": 1,
"stock": 100,
"beginTime": "2022-10-25T12:09:04",
"endTime": "2022-12-25T12:09:04"
}

可以通过postman调用

实现秒杀下单:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-v54T0KOV-1668574944846)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114192754339.png)]

下单时需要判断两点:

  • 秒杀是否开始或结束,如果尚未开始或已经结束无法下单
  • 库存是否充足,不足则无法下单

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1AtDzqAS-1668574944846)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114193239439.png)]

库存超卖问题分析:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oMPRwTiA-1668574944847)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114233014147.png)]

超卖问题是典型的多线程安全问题,针对这一问题的常见解决方案就是加锁:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-iKY8HQuU-1668574944848)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114233544182.png)]

乐观锁:

  • 版本号法
    - [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JBV8SEkv-1668574944849)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114234218883.png)]

  • CAS法(CompareAndSet)

    • [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4q4SdHyN-1668574944849)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221114234628336.png)]

乐观锁解决超卖问题:

//1.查询优惠劵
SeckillVoucher voucher = iSeckillVoucherService.getById(voucherId);
//2.判断秒杀是否开始
LocalDateTime beginTime = voucher.getBeginTime();
if (beginTime.isAfter(LocalDateTime.now())) {
    //尚未开始
    return Result.fail("活动尚未开始");
}
//3.判断秒杀是否已经结束
LocalDateTime endTime = voucher.getEndTime();
if (LocalDateTime.now().isAfter(endTime)) {
    //已结束
    return Result.fail("活动已经结束");
}
//4判断库存是否充足
if (voucher.getStock() < 1) {
    //库存不足
    return Result.fail("库存不足!");
}
//5.扣减库存
boolean success =iSeckillVoucherService.update()
        .setSql("stock =stock -1")
        .eq("voucher_id",voucherId).gt("stock",0).
        update();

if (!success) {
    //扣减失败
    return Result.fail("库存不足!");
}
//6.创建订单
VoucherOrder voucherOrder = new VoucherOrder();
//6.1 订单id
long orderId = redisIdWorker.nextId("order");
voucherOrder.setId(orderId);
//6.2 用户id
Long userId = UserHolder.getUser().getId();
voucherOrder.setUserId(userId);
//6.3 代金券id
voucherOrder.setVoucherId(voucherId);
save(voucherOrder);
// 7.返回订单id
return Result.ok(orderId);

超卖这样的线程安全问题,解决方案有哪些?

1.悲观锁:添加同步锁,让线程串行执行

  • 优点:简单粗暴
  • 缺点:性能一般

2.乐观锁:不加锁,在更新时判读是否有其它线程在修改

一人一单:

需求:修改秒杀业务,要求同一个优惠券,一个用户只能下一单

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CDASAcEo-1668575438357)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221116130925403.png)]

    @Autowired
    private ISeckillVoucherService iSeckillVoucherService;

    @Autowired
    private RedisIdWorker redisIdWorker;

    @Override
    public Result seckillVoucher(Long voucherId) {
        //1.查询优惠劵
        SeckillVoucher voucher = iSeckillVoucherService.getById(voucherId);
        //2.判断秒杀是否开始
        LocalDateTime beginTime = voucher.getBeginTime();
        if (beginTime.isAfter(LocalDateTime.now())) {
            //尚未开始
            return Result.fail("活动尚未开始");
        }
        //3.判断秒杀是否已经结束
        LocalDateTime endTime = voucher.getEndTime();
        if (LocalDateTime.now().isAfter(endTime)) {
            //已结束
            return Result.fail("活动已经结束");
        }
        //4判断库存是否充足
        if (voucher.getStock() < 1) {
            //库存不足
            return Result.fail("库存不足!");
        }
        Long userId = UserHolder.getUser().getId();
        synchronized (userId.toString().intern()){
            //获取spring事务代理对象
            IVoucherOrderService proxy = (IVoucherOrderService) AopContext.currentProxy();
            return proxy.createVoucherOrder(voucherId);
    }
}

    @Transactional
    public  Result createVoucherOrder(Long voucherId) {
            //6.一个人一单
            Long userId = UserHolder.getUser().getId();
            //6.1查询订单
            int count = query().eq("user_id", userId).eq("voucher_id", voucherId).count();
            //6.2判断是否存在
            if (count > 0) {
                //用户以及购买过
                return Result.fail("用户已经购买过一次");
            }
            //7.扣减库存
            boolean success = iSeckillVoucherService.update()
                    .setSql("stock =stock -1")
                    .eq("voucher_id", voucherId)
                    .gt("stock", 0).update();
            if (!success) {
                //扣减失败
                return Result.fail("库存不足!");
            }
            //8.创建订单
            VoucherOrder voucherOrder = new VoucherOrder();
            //8.1 订单id
            long orderId = redisIdWorker.nextId("order");
            voucherOrder.setId(orderId);
            //8.2 用户id
            voucherOrder.setUserId(userId);
            //8.3 代金券id
            voucherOrder.setVoucherId(voucherId);
            save(voucherOrder);
            // 9.返回订单id
            return Result.ok(orderId);
        }

一人一单的并发安全问题:

通过加锁可以解决在单机情况下的一人一单安全问题,但是在集群模式下就不行了。

1.我们将服务启动两份,端口分别为8081和8082:

在这里插入图片描述

2.修改nginx的conf目录下的nginx.conf文件,配置反向代理和负载均衡:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bF5Dvek3-1668574944852)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221116123310064.png)]

现在,用户请求会在这两个节点上负载均衡,再次测试下是否存在线程安全问题。

一人一单的并发安全问题:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6Ikza5ar-1668574944856)(C:\Users\20745\AppData\Roaming\Typora\typora-user-images\image-20221116125039581.png)]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/9922.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CentOS7安装MySQL8

文章目录一 前言二、Centos 7 安装 mysql8 步骤&#xff1a;1.下载MySQL官方的 Yum Repository2.安装方法一&#xff1a; 用wget 下载后安装方法二&#xff1a;下载 RMP 软件包将该软件包上传到 Linux 服务器&#xff0c;并安装。3.Navicate 远程连接配置一 前言 最近在自己的…

Python 入门基础

第一个Python程序之打印 Hello World! print("Hello World!")字符串定义的三种方式&#xff0c;type 用了检测数据类型 # 单引号定义法&#xff0c;使用单引号进行包围 name 测试 print(type(name)) # 双引号定义法 name "测试" print(type(name)) # 三…

vue3+ts组件练习(defineExpose defineEmits defineProps)

学习关键语句&#xff1a; vue3ts 组件写法 写在前面 进化到 vue3 ts 的时代&#xff0c;vue的不少语法发生了改变&#xff0c;尤其是选项式 API 变为了组合式 API 和 typescript 的使用使得从 vue2 过来的人需要尽快熟悉新的写法&#xff0c;毕竟大差不差嘛 文章最后有本文…

图像分割 - 阈值处理 - 全局阈值处理

目录 1. 介绍 2. 代码实现 3. 代码讲解 1. 介绍 当目标和背景像素的灰度分布非常不同的时候&#xff0c;可以对整个图像使用全局阈值 在大多数的应用中&#xff0c;图像之间通常存在足够的变化&#xff0c;全局阈值是一种合适的办法。所以&#xff0c;需要一种对图像做阈值…

生存分析的图你也要拼接 图形拼接r 不同的图形组合在一起

生存分析的图你也要拼接吗 因为都是ggplot体系的图表,很容易拼接,但是里面的生存分析是一个麻烦事情。因为它本身主要是survminer包出图,而这个survminer包出图并不是很稳定,但是学员自己解决了这个问题。 可以先用survminer包的arrange_ggsurvplots函数对多个生存分析图表…

CUDA By Example(五)——常量内存与事件

本章将介绍如何使用GPU上特殊的内存区域来加速应用程序的执行&#xff0c;以及如何通过事件来测量CUDA应用程序的性能。通过这些测量方法&#xff0c;你可以定量的分析对应用程序的某个修改是否会带来性能提升 文章目录常量内存光纤跟踪简介在GPU上实现光线跟踪通过常量内存来实…

[附源码]java毕业设计家庭医生系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

Java代码审计——WebGoat XML外部实体注入(XXE)

目录 前言&#xff1a; 0x01 Let’s try 0x02 代码分析 2.1 安全的代码 0x03 Modern REST framework 3.1 解题&#xff1a; 3.2 改为xml格式: 3.3 源码分析&#xff1a; 3.4 参考解决方案 0x04 Blind XXE assignment 0x05XXE DOS attack 参考文章&#xff1a; 前言…

“百花齐放、百家争鸣”,数据可视化呈现我国科学文化的发展

公共财政对文化建设的支持日益加强&#xff0c;公共文化设施不断完善&#xff0c;覆盖城乡的公共文化服务网络初步建立&#xff0c;公共文化服务理念逐步深化&#xff0c;公共文化服务能力和均等化水平逐渐提高&#xff1b;文化产业投资向发展水平较低的中西部地区倾斜&#xf…

node.js+Express框架,前端自己创建接口

目录 一、安装 1、安装node.js 2、安装Express框架 3、安装nodemon 二、写接口 三、连接数据库 1、安装&#xff1a; 2、连接数据库 3、执行 四、注意事项 1、跨域 这篇文章看完如果您觉得有所收获&#xff0c;认为还行的话&#xff0c;就点个赞收藏一下呗 一、安装…

多线程详细介绍

一、分类 创建线程的四种方法&#xff1a; &#xff08;1&#xff09;继承Thread &#xff08;2&#xff09;实现Runnable &#xff08;3&#xff09;实现Callable &#xff08;4&#xff09;线程池创建一个新的线程可以通过继承Thread类或者实现Runnable接口来实现&#xff0…

JAVA基础—面向对象

1、面向对象介绍 2、类和对象 2.1、如何定义类、得到类的对象、使用对象 2.2、类和对象的总结 2.3、定义类的补充事项—测试类与Javabean类 成员变量一般无需指定初始值&#xff0c;存在默认值。 但是局部变量必须定义初始值。 2.4、定义类的注意事项 2.4.1、驼峰模式 单词…

容器docker安装,以及paddle容器环境安装

例如 一台新的 服务器部署环境 1.环境服务器部署 第一步:查看显卡的驱动是否装好了,可以用nvidia-smi,正常显示表示已经安装 上面版本没有问题,但是下面版本驱动本本比较低,不支持11.2cuda安装,要更新驱动 docker, nvidia-docker 安装之前先确认如下三个 paddle 2.3 …

Win11 KB5019157(22000.1281)11月累积补丁推送了!

微软在最新推出的KB5019157更新补丁中修复了Microsoft Store更新等多个错误&#xff0c;但是仍然存在Direct Access问题。Win11用户安装之后即可升级至22000.1281&#xff0c;下面就来看看详细内容。 重要信息 KB5019157累积更新补丁 它解决了 Microsoft Store 的一些持续更新失…

shell中通配符的使用

shell中的通配符与正则表达式是不同的两种功能。 正则表达式一般不会在bash直接应用&#xff0c;需要使用sed、grep、awk来解释正则表达式。 通配符则可以通过bash直接解释&#xff0c;一般用作名称展开。 bash中可以使用的通配符符号包括&#xff1a;*、?、{}、[]、^ 这些…

【环境配置笔记】基于clang15搭建liunx内核代码阅读环境

环境&#xff1a; Toolchain&#xff08;yocto导出&#xff09;Linux5-15&#xff0c;arm64VS codeclangd15.0.3 1. VScode配置 在VS code / VS code server中配置以下插件&#xff1a; ClangdC/CC/C Extension PackC/C SnippetsCode RunnerCode Spell CheckercompareitDev…

拖拽页面元素+flip动画的案例

先上效果&#xff1a; 实现思路和流程&#xff1a; 基础页面布局 给每个拖动元素加上 draggable"true"ondragstart(开始拖动某个元素时)做出 对应的处理 获得操作的具体元素 给目标元素添加对应的样式 显示透明 增加虚线描边ondragover 被拖动的元素hover到目标元素…

java计算机毕业设计ssm养老管理系统-敬老院系统

项目介绍 1.登录页面要有验证码 2.在健康信息模块中有三个小模块分别是饮食信息、身体信息、医疗常识,饮食信息就是护工每天负责老人的饮食搭配包括;早中晚餐;各种菜谱、图片、注意事项等,身体信息就是老人的各项身体指标,医疗常识就是每一种病对应一种病例还有每天推送更新的关…

互联网获客经验分享(一)

现在的流量都很贵了。特别是公域的流量。在这样的情况下,我们如何从互联网上获取客户?满足我业务的需求呢。 今天为大家分享基础的方法,后续我们会不定期的分享一些互联网软件获客的经验,希望能够帮助到你。 流量的本质 不管是平台方还是用户,流量都是生命线。在这样的…

好心情心理咨询平台:独处≠孤独,独处对心理健康有多重要?

说到独处&#xff0c;有人认为那不就是自己一个人呆着嘛。 非也非也&#xff0c;独处其实不是要求远离人群&#xff0c;而是在信息与情感上与他人无交流&#xff0c;做到无社会互动&#xff0c;与人在网上聊天、打电话可都不能算是独处。 相反&#xff0c;自己一个人在图书馆…