Laravel8.5+微信小程序实现京东商城秒杀方案

一、商品秒杀涉及的知识点

  • 鉴权策略封装
  • 掊口访问频次限制
  • 小程序设计
  • 页面防抖
  • 接口调用
  • 订单创建事务使用
  • 超卖防御

二、订单库存系统方案(3种)

扣库存的集中方案

  1. 下单减库存 优点是库存和订单的强一致性,商品不会卖超,但是可能导致恶意下单,影响正常流程
  2. 支付减库存 优点是避免恶意下单,支付和库存强一致性,但是可能出现订单无法支付,商品库存不足等问题。
  3. 预扣库存 预扣库存是指用户请求之后预扣库存,生成订单,在时效内支付,否则订单失效,库存还原
  4. 代码:itshujia.com 或者wx搜:《IT书架》 回复:秒杀

三、小程序秒杀页面

  • 商品秒杀详情页面页面
  • 该商品秒杀时间未到,则上方进行提醒秒杀商品倒计时,并且立即抢购禁用(按钮置灰)

image.png

  • 商品秒杀时间开始,则把立即秒杀按钮禁用状态改为可以点击
  • 点击立即秒杀按钮要考虑页面防抖,不能重复在同一秒中重复发起N次网络请求
    image.png

四、小程序部分代码展示

  1. 小程序wxml代码
<view>商品秒杀页面</view>
<l-countdown time-type="second" time="{{expire_time}}" bind:linend="changeBtn" />
<l-card type="primary" full="{{true}}" image="{{goods.goods.goods_image}}" title="{{goods.goods.goods_name}}"><view>价格:{{goods.goods.goods_price}}</view><view><!-- <button disabled="true" bindtap="buyGoods" >抢购</button> --><l-button disabled="{{ btn_disable }}" bind:lintap="buyGoods" type="error" data-goods_id="{{ goods.goods.id }}">立即秒杀</l-button></view></l-card>
  1. 小程序js代码
// pages/goods_detail/goods_detail.js
import {debounce
} from "../../utils/util"Page({/*** 页面的初始数据*/data: {goods:{},expire_time:0,btn_disable:false},/*** 生命周期函数--监听页面加载*/onLoad: function (e) {//商品idlet goods_id = e.goods_id;this.getGoodsDetail(goods_id)},//获取商品详情getGoodsDetail(goods_id){var token = wx.getStorageSync('token')wx.request({url: 'http://www.zfw.com/api/v1/goods_detail?goods_id='+goods_id,header: {'Authorization': `Bearer ${token}`},success: res => {//当前的时间戳let now_time = Math.round(new Date().getTime() / 1000).toString();let expire_time = res.data.data.start_time-now_time;this.setData({goods:res.data.data,expire_time})if(expire_time > 0){this.setData({btn_disable:true})}console.log(this.data.goods)}})},//立即抢购   debounce此方法是引入的util工具类的页面防抖函数buyGoods:debounce(function (e) {let goods_id = e[0].currentTarget.dataset.goods_idvar token = wx.getStorageSync('token')wx.request({url: 'http://www.zfw.com/api/v1/snap_up',header: {'Authorization': `Bearer ${token}`},method:"POST",data:{goods_id},success: res => {let code = res.statusCode.toString()if (!code.startsWith('2')){wx.showToast({title: '异常!',icon:1})}if(res.data.errorCode == 0){wx.redirectTo({url: '/pages/order/order',})}else{wx.showToast({title: res.data.msg,})}console.log(res.data)}})}),changeBtn(){this.setData({btn_disable:false})}
})
  1. 小程序json代码,引入的自定义组件
{"usingComponents": {"l-card":"/miniprogram_npm/lin-ui/card","l-button":"/miniprogram_npm/lin-ui/button","l-countdown":"/miniprogram_npm/lin-ui/countdown"}
}

五、后台业务逻辑

要考虑订单超卖,这次代码使用的redis队列实现的

<?phpnamespace App\Http\Controllers\Api;use App\Http\Controllers\Controller;
use App\Models\Goods;
use Illuminate\Http\Request;
use App\Models\ActivityGoods;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Redis;
use Lcobucci\JWT\Exception;class GoodsController extends Controller
{//商品秒杀列表public function activityList(){$result = ActivityGoods::with(['goods'])->get();return response()->json(['errorCode' => 0, 'data' => $result, 'msg' => '查询成功']);}//商品秒杀列表public function goodsDetail(Request $request){$goods_id = $request->get('goods_id');$result = ActivityGoods::with(['goods'])->where('goods_id',$goods_id)->first();return response()->json(['errorCode' => 0, 'data' => $result, 'msg' => '查询成功']);}//同步库存public function syncStock(){//查出所有参与秒杀活动列表$result = ActivityGoods::with(['goods'])->get()->toArray();//进行把参与秒杀的商品写入到数据库foreach ($result as $val){//生成对应商品库存队列$goods = "activity_goods_".$val['goods_id'];for ($i=0; $i < $val['sku_nums']; $i++) {Redis::lpush($goods, 1);}}}//校验库存public function checkStock(Request $request){//获取token$token = explode(' ',$request->header('authorization'))[1];//进行查看$userInfo = Cache::get($token);//抢购用户id$userID = $userInfo->id;//商品id$goodsID = $request->input("goods_id");//对应商品库存队列$goods = "activity_goods_".$goodsID;//对应商品抢购成功用户集合 {1,3,4}$robSuccessUser = "success_user".$goodsID;//进行判断当前用户是否在抢成功的队列里面$result = Redis::sismember($robSuccessUser,$userID);//如果你在这里面,就抢完了if ($result) {//如果抢购成功 返回状态码,进行下单return response()->json(['errorCode' => 20000, 'data' => '', 'msg' => '已经抢购过了']);}//减库存,把队列里面的数据从左边 头$count = Redis::lpop($goods);if (!$count) {//如果抢购成功 返回状态码,进行下单return response()->json(['errorCode' => 20001, 'data' => '', 'msg' => '已经抢光了哦']);}//把当前这个秒杀的uid存储到中奖的队列里set$success = Redis::sadd($robSuccessUser, $userID);if(!$success){//已经在成功队列里了,加回库存,防止的是同个用户并发请求Redis::lpush($goods, 1);//如果抢购成功 返回状态码,进行下单return response()->json(['errorCode' => 20002, 'data' => '', 'msg' => '已经抢购过了']);}//如果抢购成功 返回状态码,进行下单return response()->json(['errorCode' => 0, 'data' => '', 'msg' => '秒杀成功']);}//创建订单public function createOrder(Request $request){//获取token$token = explode(' ',$request->header('authorization'))[1];//进行查看$userInfo = Cache::get($token);//抢购用户id$userID = $userInfo->id;//商品id$goodsID = $request->input("goods_id");//对应商品抢购成功用户集合 {1,3,4}$robSuccessUser = "success_user".$goodsID;//进行判断当前用户是否在抢成功的队列里面$result = Redis::sismember($robSuccessUser,$userID);//如果你在这里面,就抢完了if (!$result) {//如果抢购成功 返回状态码,进行下单return response()->json(['errorCode' => 20003, 'data' => '', 'msg' => '手慢了!']);}DB::beginTransaction();try{//减库存//生成订单DB::commit();//下单成功,跳转支付页面return response()->json(['errorCode' => 0, 'data' => '', 'msg' => '下单成功!']);}catch (\Exception $e){DB::rollBack();}}
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JVM:即时编译器,C2 Compiler,堆外内存排查

1&#xff0c;即时编译器 1.1&#xff0c;基本概念 常见的编译型语言如C&#xff0c;通常会把代码直接编译成CPU所能理解的机器码来运行。而Java为了实现“一次编译&#xff0c;处处运行”的特性&#xff0c;把编译的过程分成两部分&#xff0c;首先它会先由javac编译成通用的…

屏幕分辨率|尺寸|颜色深度指纹修改

一、前端通过window.screen接口获取屏幕分辨率 尺寸 颜色深度&#xff0c;横屏竖屏信息。 二、window.screen c接口实现&#xff1a; 1、third_party\blink\renderer\core\frame\screen.idl // https://drafts.csswg.org/cssom-view/#the-screen-interface[ExposedWindow ] …

【C#设计模式(15)——命令模式(Command Pattern)】

前言 命令模式的关键通过将请求封装成一个对象&#xff0c;使命令的发送者和接收者解耦。这种方式能更方便地添加新的命令&#xff0c;如执行命令的排队、延迟、撤销和重做等操作。 代码 #region 基础的命令模式 //命令&#xff08;抽象类&#xff09; public abstract class …

Linux基础项目包含(DNS,FTP,Samba)

今天我们来实现一个linux的基础项目&#xff0c;多做项目可以加深我们对每个服务的掌握程度&#xff0c;能更加的熟悉各个服务的使用&#xff0c;以及每个服务之间如何相互应用&#xff0c;我们可以有一个清晰的认知&#xff0c;那么接下来我们就开始项目的实操。 一、 项目背…

鸿蒙面试 --- 性能优化

性能优化可以从三个方面入手 感知流畅、渲染性能、运行性能 感知流畅 在应用开发中&#xff0c;动画可以为用户界面增添生动、流畅的交互效果&#xff0c;提升用户对应用的好感度。然而&#xff0c;滥用动画也会导致应用性能下降&#xff0c;消耗过多的系统资源&#xff0c;…

UWB数字钥匙安全测距和场景应用

1. CCC数字钥匙 2021年7月CCC将UWB定义为第三代数字钥匙的核心技术&#xff0c;并发布CCC R3&#xff08;第三代数字钥匙&#xff09;规范。 CCC R3是基于NFC/BLE/UWB作为基础的无线电技术的使用&#xff0c;该系统采用非对称密码技术对车辆和设备进行相互签名认证&#xff0…

使用Compose Multiplatform开发跨平台的Android调试工具

背景 最近对CMP跨平台很感兴趣&#xff0c;为了练手&#xff0c;在移动端做了一个Android和IOS共享UI和逻辑代码的天气软件&#xff0c;简单适配了一下双端的深浅主题切换&#xff0c;网络状态监测&#xff0c;刷新调用振动器接口。 做了两年多车机Android开发&#xff0c;偶…

ubuntu22开机自动登陆和开机自动运行google浏览器自动打开网页

一、开机自动登陆 1、打开settings->点击Users 重启系统即可自动登陆桌面 二、开机自动运行google浏览器自动打开网页 1、安装google浏览器 sudo wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb sudo dpkg -i ./google-chrome-stable…

前端学习笔记之文件下载(1.0)

因为要用到这样一个场景&#xff0c;需要下载系统的使用教程&#xff0c;所以在前端项目中就提供了一个能够下载系统教程的一个按钮&#xff0c;供使用者进行下载。 所以就试着写一下这个功能&#xff0c;以一个demo的形式进行演示&#xff0c;在学习的过程中也发现了中文路径…

FPGA实现GTP光口视频转USB3.0传输,基于FT601+Aurora 8b/10b编解码架构,提供3套工程源码和技术支持

目录 1、前言工程概述免责声明 2、相关方案推荐我已有的所有工程源码总目录----方便你快速找到自己喜欢的项目我这里已有的 GT 高速接口解决方案本博已有的FPGA驱动USB通信方案 3、工程详细设计方案工程设计原理框图输入Sensor之-->OV5640摄像头输入Sensor之-->芯片解码的…

借助nntools的模型训练(图像去噪任务, PyTorch)

在跑通开源代码的过程中&#xff0c;一直难以找到很合适的代码框架或代码模板&#xff0c;尤其对于初学者而言是不小的挑战。本人在跑代码的过程中&#xff0c;发现在GitHub上有一个辅助训练过程的模板。这篇文章主要记录它的使用方法&#xff0c;并标记来源。 代码均在jupyter…

C# 常量

文章目录 前言一、整数常量&#xff08;一&#xff09;合法与非法实例对比&#xff08;二&#xff09;不同进制及类型示例 二、浮点常量三、字符常量四、字符串常量五、定义常量 前言 在 C# 编程的世界里&#xff0c;常量是一类特殊的数据元素&#xff0c;它们如同程序中的 “定…

PHP 生成分享海报

因为用户端有多个平台&#xff0c;如果做分享海报生成&#xff0c;需要三端都来做&#xff0c;工作量比较大。 所以这个艰巨的任务就光荣的交给后端了。经过一定时间的研究和调试&#xff0c;最终圆满完成了任务&#xff0c;生成分享海报图片实现笔记如下。 目录 准备字体文件…

探索Python词云库WordCloud的奥秘

文章目录 探索Python词云库WordCloud的奥秘1. 背景介绍&#xff1a;为何选择WordCloud&#xff1f;2. WordCloud库简介3. 安装WordCloud库4. 简单函数使用方法5. 应用场景示例6. 常见Bug及解决方案7. 总结 探索Python词云库WordCloud的奥秘 1. 背景介绍&#xff1a;为何选择Wo…

git 学习笔记

目录 一、git 前期配置准备 1、托管平台的账号注册&#xff08;以gitee码云为demo&#xff09; 2、本地个人电脑配置 &#xff08;1&#xff09;配置用户属性 ​编辑 &#xff08;2&#xff09;配置SSH密钥 二、git 工作流程图 三、git 提交命令 &#xff08;1&#x…

uniapp中使用uni-forms实现表单管理,验证表单

前言 uni-forms 是一个用于表单管理的组件。它提供了一种简化和统一的方式来处理表单数据&#xff0c;包括表单验证、字段绑定和提交逻辑等。使用 uni-forms可以方便地创建各种类型的表单&#xff0c;支持数据双向绑定&#xff0c;可以与其他组件及API进行良好的集成。开发者可…

_C#_串口助手_字符串拼接缺失问题(未知原理)

最近使用WPF开发串口助手时&#xff0c;遇到一个很奇怪的问题&#xff0c;无论是主线程、异步还是多线程&#xff0c;当串口接收速度达到0.016s一次以上&#xff0c;就会发生字符串缺失问题并且很卡。而0.016s就一切如常&#xff0c;仿佛0.015s与0.016s是天堑之隔。 同一份代码…

JVM(JAVA虚拟机)内存溢出导致内存不足,Java运行时环境无法继续

1、先贴出服务最后打印出来的日志&#xff0c;意思就是给虚拟机分配的内存被用完了&#xff0c;没有可用的内存了&#xff0c;服务运行不了了&#xff0c;被动停服了。详细的日志记录在了/home/user/zx/tomcat/apache-tomcat-8.5.82/bin/hs_err_pid147951.log文件里。 Java Ho…

OGRE 3D----3. OGRE绘制自定义模型

在使用OGRE进行开发时,绘制自定义模型是一个常见的需求。本文将介绍如何使用OGRE的ManualObject类来创建和绘制自定义模型。通过ManualObject,开发者可以直接定义顶点、法线、纹理坐标等,从而灵活地构建各种复杂的几何体。 Ogre::ManualObject 是 Ogre3D 引擎中的一个类,用…

Spring Boot英语知识网站:架构与开发

4系统概要设计 4.1概述 本系统采用B/S结构(Browser/Server,浏览器/服务器结构)和基于Web服务两种模式&#xff0c;是一个适用于Internet环境下的模型结构。只要用户能连上Internet,便可以在任何时间、任何地点使用。系统工作原理图如图4-1所示&#xff1a; 图4-1系统工作原理…