redis实现高并发下的抢购/秒杀功能

常规写法:

查询出对应商品的库存,看是否大于0,然后执行生成订单等操作,但是在判断库存是否大于0处,如果在高并发下就会有问题,导致库存量出现负数

redis的解决方案:

1,<?php
inlcude_onceA './include/db.php';
$redis = new Redis();
$resid->connect('127.0.0.1', 6379);
$reds_name = 'miaosha';
$db = DB::getInstance();
//死循环
while(1) {
 //从队列最左侧取出一个值来
 $user = $redis->lPop($redis_name);
 //然后判断这个值是否存在
 if (!$user || $user == 'nil') {
   sleep(2);
   continue;
 }
//切割出时间, uid
$user_arr = explode('%', $user);
$insert_data = array(
    'uid' => $user_arr[0],
    'time_stamp' => $use_arr[1]]
);
 //保存到数据库中
 $res = $db->insert('redis_queue', $insert_data);:
 //数据库插入失败时回滚
  if (!$res) {
      $redis->rPop($redis_name, $user);
  }
 }
 //释放redis
 $redis->close();
}

 

2.

</php
//首先,加载一个Reids组件,
$redis = new Redis();
$resid->connect('127.0.0.1', 6379);
$reds_name = 'miaosha';
for ($i = 0;$i < 100;$i++) {
    $uid = rand(1000000, 9999999);
}
 
//接收用户的id,
//$uid = $_GET['uid'];
//获取redis里面已有的数量
$num = 10;
//如果当前人数少于10的时候,则加入这个队列
if ($redis->lLen($resid_name) < 10) {
    $redis->rPush($redis_name . $uid . '%' . microtime());
    echo $uid . '秒杀成功';        
}else{
    //如果当前人数已经达到10人,责任返回秒杀已完成
        echo '秒杀已结束';
}
$redis->close();
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Redis可以通过使用事务或Lua脚本来实现高并发下的抢购/秒杀功能。 1. 使用事务:抢购/秒杀的过程可以看做是一个先检查库存是否充足,再扣减库存的过程。使用Redis的事务可以保证这个过程是原子性的,即要么全部成功,要么全部失败。 具体实现方法如下: - 使用MULTI命令开启一个事务。 - 使用WATCH命令对库存进行监视(即设置监视器)。 - 使用GET命令获取当前库存。 - 判断库存是否充足,如果充足,则使用DECRBY命令扣减库存。 - 使用EXEC命令提交事务,如果提交成功,则说明扣减成功,否则说明被其他线程抢先扣减了库存。 示例代码如下: ```python def decrease_stock(redis_conn, stock_key): with redis_conn.pipeline() as pipeline: while True: try: pipeline.watch(stock_key) stock = int(pipeline.get(stock_key)) if stock > 0: pipeline.multi() pipeline.decr(stock_key) pipeline.execute() return True else: return False except WatchError: continue ``` 2. 使用Lua脚本:Lua脚本可以在Redis端原子性地执行多个命令,可以减少网络开销和锁竞争的问题。 具体实现方法如下: - 编写一个Lua脚本,该脚本首先使用GET命令获取当前库存,如果库存充足,则使用DECRBY命令扣减库存,否则返回0。 - 在Python中使用Redis的EVAL命令执行该Lua脚本。 示例代码如下: ```python def decrease_stock(redis_conn, stock_key): script = """ local stock = tonumber(redis.call('GET', KEYS[1])) if stock > 0 then redis.call('DECRBY', KEYS[1], 1) return 1 else return 0 end """ result = redis_conn.eval(script, 1, stock_key) return bool(result) ``` 以上两种方法都可以实现高并发下的抢购/秒杀功能,但是使用Lua脚本的方法效率更高,因为在Redis端执行命令可以减少网络开销和锁竞争的问题。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值