PHP 使用redis实现秒杀

使用redis队列,因为pop操作是原子的,即使有很多用户同时到达,也是依次执行,推荐使用(mysql事务在高并发下性能下降很厉害,文件锁的方式也是)

先将商品库存如队列

[php] view plain copy

  1. <?php
  2. $store=1000;
  3. $redis=new Redis();
  4. $result=$redis->connect(‘127.0.0.1‘,6379);
  5. $res=$redis->llen(‘goods_store‘);
  6. echo $res;
  7. $count=$store-$res;
  8. for($i=0;$i<$count;$i++){
  9. $redis->lpush(‘goods_store‘,1);
  10. }
  11. echo $redis->llen(‘goods_store‘);
  12. ?>

抢购、描述逻辑

[php] view plain copy

    1. <?php
    2. $conn=mysql_connect("localhost","big","123456");
    3. if(!$conn){
    4. echo "connect failed";
    5. exit;
    6. }
    7. mysql_select_db("big",$conn);
    8. mysql_query("set names utf8");
    9. $price=10;
    10. $user_id=1;
    11. $goods_id=1;
    12. $sku_id=11;
    13. $number=1;
    14. //生成唯一订单号
    15. function build_order_no(){
    16. return date(‘ymd‘).substr(implode(NULL, array_map(‘ord‘, str_split(substr(uniqid(), 7, 13), 1))), 0, 8);
    17. }
    18. //记录日志
    19. function insertLog($event,$type=0){
    20. global $conn;
    21. $sql="insert into ih_log(event,type)
    22. values(‘$event‘,‘$type‘)";
    23. mysql_query($sql,$conn);
    24. }
    25. //模拟下单操作
    26. //下单前判断redis队列库存量
    27. $redis=new Redis();
    28. $result=$redis->connect(‘127.0.0.1‘,6379);
    29. $count=$redis->lpop(‘goods_store‘);
    30. if(!$count){
    31. insertLog(‘error:no store redis‘);
    32. return;
    33. }
    34. //生成订单
    35. $order_sn=build_order_no();
    36. $sql="insert into ih_order(order_sn,user_id,goods_id,sku_id,price)
    37. values(‘$order_sn‘,‘$user_id‘,‘$goods_id‘,‘$sku_id‘,‘$price‘)";
    38. $order_rs=mysql_query($sql,$conn);
    39. //库存减少
    40. $sql="update ih_store set number=number-{$number} where sku_id=‘$sku_id‘";
    41. $store_rs=mysql_query($sql,$conn);
    42. if(mysql_affected_rows()){
    43. insertLog(‘库存减少成功‘);
    44. }else{
    45. insertLog(‘库存减少失败‘);
    46. }   
        1. --
        2. -- 数据库: `big`
        3. --
        4. -- --------------------------------------------------------
        5. --
        6. -- 表的结构 `ih_goods`
        7. --
        8. CREATE TABLE IF NOT EXISTS `ih_goods` (
        9. `goods_id` int(10) unsigned NOT NULL AUTO_INCREMENT,
        10. `cat_id` int(11) NOT NULL,
        11. `goods_name` varchar(255) NOT NULL,
        12. PRIMARY KEY (`goods_id`)
        13. ) ENGINE=MyISAM  DEFAULT CHARSET=utf8 AUTO_INCREMENT=2 ;
        14. --
        15. -- 转存表中的数据 `ih_goods`
        16. --
        17. INSERT INTO `ih_goods` (`goods_id`, `cat_id`, `goods_name`) VALUES
        18. (1, 0, ‘小米手机‘);
        19. -- --------------------------------------------------------
        20. --
        21. -- 表的结构 `ih_log`
        22. --
        23. CREATE TABLE IF NOT EXISTS `ih_log` (
        24. `id` int(11) NOT NULL AUTO_INCREMENT,
        25. `event` varchar(255) NOT NULL,
        26. `type` tinyint(4) NOT NULL DEFAULT ‘0‘,
        27. `addtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
        28. PRIMARY KEY (`id`)
        29. ) ENGINE=MyISAM DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;
        30. --
        31. -- 转存表中的数据 `ih_log`
        32. --
        33. -- --------------------------------------------------------
        34. --
        35. -- 表的结构 `ih_order`
        36. --
        37. CREATE TABLE IF NOT EXISTS `ih_order` (
        38. `id` int(11) NOT NULL AUTO_INCREMENT,
        39. `order_sn` char(32) NOT NULL,
        40. `user_id` int(11) NOT NULL,
        41. `status` int(11) NOT NULL DEFAULT ‘0‘,
        42. `goods_id` int(11) NOT NULL DEFAULT ‘0‘,
        43. `sku_id` int(11) NOT NULL DEFAULT ‘0‘,
        44. `price` float NOT NULL,
        45. `addtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
        46. PRIMARY KEY (`id`)
        47. ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT=‘订单表‘ AUTO_INCREMENT=1 ;
        48. --
        49. -- 转存表中的数据 `ih_order`
        50. --
        51. -- --------------------------------------------------------
        52. --
        53. -- 表的结构 `ih_store`
        54. --
        55. CREATE TABLE IF NOT EXISTS `ih_store` (
        56. `id` int(11) NOT NULL AUTO_INCREMENT,
        57. `goods_id` int(11) NOT NULL,
        58. `sku_id` int(10) unsigned NOT NULL DEFAULT ‘0‘,
        59. `number` int(10) NOT NULL DEFAULT ‘0‘,
        60. `freez` int(11) NOT NULL DEFAULT ‘0‘ COMMENT ‘虚拟库存‘,
        61. PRIMARY KEY (`id`)
        62. ) ENGINE=InnoDB  DEFAULT CHARSET=utf8 COMMENT=‘库存‘ AUTO_INCREMENT=2 ;
        63. --
        64. -- 转存表中的数据 `ih_store`
        65. --
        66. INSERT INTO `ih_store` (`id`, `goods_id`, `sku_id`, `number`, `freez`) VALUES
        67. (1, 1, 11, 500, 0);
时间: 2024-11-05 07:52:50

PHP 使用redis实现秒杀的相关文章

.NetCore+Jexus代理+Redis模拟秒杀商品活动

开篇叙 本篇将和大家分享一下秒杀商品活动架构,采用的架构方案正如标题名称.NetCore+Jexus代理+Redis,由于精力有限所以这里只设计到商品添加,抢购,订单查询,处理队列抢购订单的功能:有不足或者不够详细的还请见谅,顺手点个推荐也不错: a. 秒杀流程 b. 封装StackExchange.Redis的使用类 c. Ubuntu16.04上使用Jexus搭建代理完成分布式部署 d. NetCore写实时监控队列服务 秒杀架构设计图︿( ̄︶ ̄)︿三幅 1. 一般业务性架构 2. 后端分布

thinkphp+redis实现秒杀功能

1,安装redis,根据自己的php版本安装对应的redis扩展(此步骤简单的描述一下) 1.1,安装 php_igbinary.dll,php_redis.dll扩展此处需要注意你的php版本如图: 1.2,php.ini文件新增 extension=php_igbinary.dll;extension=php_redis.dll两处扩展 ok此处已经完成第一步redis环境搭建完成看看phpinfo 2,项目中实际使用redis 2.1,第一步配置redis参数如下,redis安装的默认端口

redis优化秒杀系统

用redis提高基于mss的秒杀系统 使用背景: 普通的基于mss框架的系统在并发量不是很高的情况下,对redis的需求不是很高.redis在系统中的角色相当于一个对象缓存器,在高并发的系统中(比如秒杀系统),在某一刻对数据库中的一条数据可能是成千上万的用户同时去访问,系统的用户体验度直接受到数据库的性能的影响.为了保证数据的完整性,用户只能串行访问数据库中的某一条记录.redis则是把记录对应的对象序列化存储在自身的容器中,减少数据库的压力.废话不多说,接下来简单介绍redis的使用. red

借助Redis做秒杀和限流的思考

最近群里聊起秒杀和限流,我自己没有做过类似应用,但是工作中遇到过更大的数据和并发. 于是提出了一个简单的模型: var count = rds.inc(key); if(count > 1000) throw "已抢光!" 借助Redis单线程模型,它的inc是安全的,确保每次加一,然后返回加一后的结果.如果原来是234,加一了就是235,返回的一定是235,在此中间,不会有别的请求来打断从而导致返回236或者其它. 其实我们可以理解为inc的业务就是占坑排队,每人占一个坑,拿到

redis实现秒杀demo

代码 package com.prosay.redis; import java.util.List; import redis.clients.jedis.Jedis; import redis.clients.jedis.Transaction; public class SaleRunable implements Runnable{ String productKey="iphone8";//监视的key 当前秒杀商品的数量 Jedis jedis = new Jedis(&q

秒杀思路: yii2加Redis实现秒杀

整理一些思路 PHP7, yii2加Redis怎么实现秒杀 一个list列表存放秒杀的商品 一个list列表存放秒杀的客户信息 首先读取秒杀商品到商品列表, 然后等待客户秒杀. 每次客户请求, 检查用户列表与商品数量是否大于等于. 如果是就提示已经抢完.下次再来. (这里可以选择方案 是用户下单就减少还是付款再减少) 检查商品的剩余数量是否大于0, 是就减一,并加入客户到列表里面. 否就提示已经抢完.下次再来. (这里可以选择方案 是用户下单就减少还是付款再减少) 每次客户请求, 加入客户到列表

Redis+PHP秒杀设计

讲师介绍: 稳定性建设: 如何做到服务9999的稳定性即每年不可用时间不到1min 高性能系统: 滴滴运营广告系统即并发量很大 社招面试: 秒杀系统的设计与实现,其本质高并发高可用,这两方面的技术可直接决定公司技术水平,提现个人技术实力 概要: 基本知识和原理 减而治之 cdn原理: 减少读的压力,如订单详情页下发到不同地方的cdn节点,访问加速,回源减少 Nginx限流: 请求到达服务端(即接入层)如何做过载保护 异步队列概念: 如通过异步方式创建订单 分而治之 Nginx负载均衡: 流量到达

使用 redis 减少 秒杀库存 超卖思路

由于数据库查询的及插入的操作 耗费的实际时间要耗费比redis 要多, 导致 多人查询时库存有,但是实际插入数据库时却超卖 redis 会有效的减少相关的延时,对于并发量相对较少的 可以一用 1 public function buy($goods_id = 0){ 2 if(!$goods_id){ 3 die("商品不存在!"); 4 } 5 $redis = new Redis(); 6 $redis->connect('127.0.0.1',6379); 7 $stock

电商项目中使用Redis实现秒杀功能

参与过抢购活动就知道,很明显的一点是商即便商品实际没有了也是可以下单成功的,但是在支付的时候会提示你商品没有了. 实现原理:list双向链表 使用redis队列,因为pop操作是原子的,即使有很多用户同时到达,也是依次执行.(mysql事务在高并发下性能下降很厉害,文件锁的方式也是). 此处用到了Redis中的链表(list)数据类型: '栈':从链表的头部添加元素,先进后出 '队列':从链表的尾部添加元素,先进先出 redis保存数据时都有key和value,key和value要么是Strin