素材牛VIP会员
php+redis抢购功能,并发问题
 雪***儿  分类:PHP代码  人气:1031  回帖:5  发布于6年前 收藏

今天尝试写并发抢购功能,再网上找了相关资料后,实现了如下:

<?php
#开始抢购,检测库存
$inventoryData = $this->redis->get($inventoryKey);
if ($inventoryData == null) {
    //库存不足
    return '-1';
}

#解析json
$inventoryData = json_decode($inventoryData, true);
#检测库存
if ($inventoryData['inventory'] == 0) {
    //库存不足
    return '-1';
}
//开启事务
$this->redis->watch($inventoryKey);
//事务开始
$this->redis->multi();
//将抢购数据加入redis
$this->redis->lPush('kill_goods_' . $data['goods_id'], json_encode($data));
$goodsData = ['id' => $data['goods_id'], 'inventory' => $inventoryData['inventory'] - 1];
$this->redis->set($inventoryKey, json_encode($goodsData));
$result = $this->redis->exec();
if ($result) {
    file_put_contents('1.txt', $inventoryData['inventory'] . PHP_EOL, FILE_APPEND);
    return 1;
}
return '-1';

经过并发测试:始终都会有重复的,请各位大佬讲解讲解

 标签:redisphp

讨论这个帖子(5)垃圾回帖将一律封号处理……

Lv5 码农
Am***ze JS工程师 6年前#1

用原子自减只能解决redis本身内的并发问题,但程序中并发处理也有问题。像Nine所说的一样,会存在多条访问全都通过前面的检测。必须在php程序内做原子限制。像一些高并发下的活动页面,根据实际业务做法有很多。像楼主的业务逻辑,我觉得首先可以让库存的key自减,然后直接根据返回key的值直接判断是否小于或者等于0。

Lv5 码农
疯***斯 职业无 6年前#2

使用lua脚本来实现扣库存。lua脚本再redis中以原子方式执行,就不会出现这种问题了。

Lv3 码奴
上***水 职业无 6年前#3

库存做原子自减

Lv3 码奴
威***军 职业无 6年前#4

我觉得这个问题可以不用事务解决的啊,还有原子自减string也是可以减为负的啊,难道负的库存也合理么?这个东西用list完美解决,根本不需要事务啊

Lv5 码农
Co***ht 软件测试工程师 6年前#5

这种方式其实并没有解决并发的问题,举个简单的例子:我们假设现在还剩下最后一个产品,两个用户进来之后,都通过了"库存检测"这道门槛,是不是存在着这样一种可能,A用户先开启并完成事务,B紧接着再完成?

 文明上网,理性发言!   😉 阿里云幸运券,戳我领取