redis decr 防止超卖_php使用redis的几种常见方式和用法

本文介绍了Redis在PHP中的多种应用场景,包括利用Redis进行简单的字符串缓存以优化SQL查询,通过列表模拟简单队列实现任务批量处理,如邮件发送和商品抢购,利用watch+multi实现乐观锁确保数据一致性,以及使用set实现悲观锁保证资源的互斥访问。此外,还展示了发布订阅功能在消息传递中的应用。
摘要由CSDN通过智能技术生成

一、简单的字符串缓存

比如针对一些sql查询较慢,更新不频繁的数据进行缓存。

<?php
 
$redis = new Redis();
$redis->connect('127.0.0.1', 6379, 60);
 
$sql = 'select * from tb_order order by id desc limit 10';
//伪代码,从数据库中获取数据
$data = $db->query($sql);
$data = json_encode($data, JSON_UNESCAPED_UNICODE);
$key = md5($sql);
//缓存数据
$redis->set($key, $value, 60);
 
//获取数据
$data = $redis->get($key);
print_r(json_decode($data, true));

二、通过列表模拟简单队列

比如我们需要批量的发送邮件,可以把发送邮件的任务存入队列中,然后启多个php脚本从队列中读取任务去发送邮件。

也可以用来处理商品秒杀,用户点击抢购时,把一个个的用户抢购任务放入队列中,串行化处理,判断队列数量,防止超卖的发生。

<?php
 
$redis = new Redis();
$redis->connect('127.0.0.1', 6379, 60);
 
//循环的把发送1000条邮件任务插入队列
for ($ix = 0; $ix < 1000; $ix++) {
    $redis->lPush('send_email_queue', json_encode([
        'id' => $ix,
        'send' => 'xxx@qq.com',
        'receive' => 'yyy@qq.com',
        'title' => 'xxx',
        'body' => 'xxx',
    ]));
}
 
sleep(3);
 
//从队列中取任务,执行任务
while ($count = $redis->lLen('send_email_queue')) {
    echo "当前任务队列数 {$count} <br>";
    $task = $redis->rpop('send_email_queue');
    $task = json_decode($task, true);
    //伪代码,发送邮件
    $mailer->send($task['send'], $task['receive'], $task['title'], $task['body']);
    echo "任务 {$task['id']} 邮件发送成功<br>";
}

三、通过watch + multi 来实现乐观锁

乐观锁,顾名思义,乐观的认为数据不会被修改,只有当更新时才去判断数据是否被修改过,通常用版本号或时间戳来实现。

redis中通过watch和multi来实现,watch会监视给定的key是否发生更改,当exec的时候如果监视的key发生过改变,则整个事务会失败。

当然我们可以调用多次watch监视多个key。

<?php
 
$redis = new Redis();
$redis->connect('127.0.0.1', 6379, 60);
 
//设置商品的库存数为100
$redis->set('goods_stock_nums', 100);
//监视该key
$redis->watch('goods_stock_nums');
 
//开启事务
$redis->multi();
 
//修改库存数
$redis->decr('goods_stock_nums');
 
//提交事务,如果在此期间有其他请求修改了该key,那么事务会失败
if ($redis->exec()) {
    echo '抢购成功';
} else {
    echo '数据错误,请重新再试';
}

四、使用 set 来实现悲观锁

悲观锁,顾名思义,悲观的认为数据总是会被修改,所以在操作前都会先加上锁,操作完后,再释放锁。

<?php
 
function getRedis()
{
    $redis = new Redis();
    $redis->connect('127.0.0.1', 6379, 60);
    return $redis;
}
 
function lock($key, $random)
{
    $redis = getRedis();
    return $redis->set($key, $random, ['nx', 'ex' => 3]);
}
 
function unlock($key, $random)
{
    $redis = getRedis();
    //使用lua脚本保证原子性
    $script = 'if redis.call("get",KEYS[1]) == ARGV[1] then return redis.call("del",KEYS[1]) else return 0 end';
    return $redis->eval($script, [$key, $random], 1);
}
 
function decrGoodsStockNums()
{
    $redis = getRedis();
 
    //获取商品库存数
    $ret = $redis->get('goods_stock_nums');
 
    if ($ret === false) {
        return false;
    }
 
    if ($ret <= 0) {
        return false;
    }
 
    $random = mt_rand();
    //先获取锁
    if (lock('goods_stock_nums_lock', $random)) {
        //修改库存数
        $redis->decr('goods_stock_nums');
 
        //释放锁
        unlock('goods_stock_nums_lock', $random);
        return true;
    } else {
        usleep(100);
        decrGoodsStockNums();
    }
}
 
decrGoodsStockNums();

五、使用 publish + subscribe 完成发布和订阅

发布代码:

<?php
 
$redis = new Redis();
$redis->pconnect('127.0.0.1', 6379);
 
$ix = 0;
//发布内容
while (true) {
    $redis->publish('news', json_encode([
        'title' => '我是新闻标题' . $ix,
        'content' => '我是新闻内容' . $ix,
        'time' => date('Y-m-d H:i:s'),
    ]));
    $ix++;
    sleep(1);
}

订阅代码:

<?php
 
$redis = new Redis();
$redis->pconnect('127.0.0.1', 6379);
 
//订阅内容
$redis->subscribe(['news'], function ($redis, $channel, $msg) {
    $msg = json_decode($msg, true);
    echo "标题: {$msg['title']} 内容: {$msg['content']} 时间: {$msg['time']} <br>";
});
  

以上内容希望帮助到大家,很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家,需要

PHP进阶架构师>>>视频、面试文档免费获取​shimo.im
3e8b02b316551c22631c0d2d0afc2093.png

或 者关注咱们下面的知乎专栏

PHP大神进阶​zhuanlan.zhihu.com
b4777f04831e4086e2052b1f0bc0faac.png
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
PHP Redis超卖是指在使用PHPRedis实现商城秒杀功能时,由于并发操作导致商品库存被多次减少,最终超出了实际库存数量。具体来说,对于秒杀活动,当多个用户同时请求购买同一件商品时,如果不采取合适的措施,可能会导致库存被重复减少,从而造成超卖的情况。 在给出解决方案之前,我们先了解下引用内容中的两种解决超卖问题的方法: 引用中的方法是使用Redis的原子性的递增递减操作来处理超卖问题。首先,从Redis中取出商品的库存数量,然后使用Redisdecr操作对库存进行减1操作。如果减1后的库存小于等于0,表示商品已经售罄,抛出异常。接下来,使用MySQL的事务对商品库存进行减1操作,并将订单信息写入数据库。如果事务执行成功,则提交事务;如果事务执行失败,则回滚事务,并将库存恢复到之前的值。 引用中的方法是使用malkusch/lock库来实现加锁操作。首先,通过传入一个Predis实例创建一个锁,并设置锁的释放时间为10秒。然后,在加锁的代码块内执行秒杀逻辑,包括从数据库获取库存数量并进行减1操作,并将订单信息写入数据库。最后,释放锁。 为了解决超卖问题,可以综合以上两种方法来进行处理。首先,使用Redisdecr操作对库存进行减1操作,并判断减1后的库存是否小于等于0,如果是则表示商品已经售罄,抛出异常。接下来,使用malkusch/lock库进行加锁操作,确保在并发情况下只有一个请求能够执行秒杀逻辑。在加锁的代码块内,执行MySQL的事务操作,包括对商品库存进行减1操作和将订单信息写入数据库。如果事务执行成功,则提交事务;如果事务执行失败,则回滚事务,并将库存恢复到之前的值。 综上所述,通过结合Redis的原子性操作和加锁操作,可以有效地解决PHP Redis超卖问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [php+redis实现商城秒杀功能](https://download.csdn.net/download/weixin_38680247/12959117)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [php 解决超卖几种方案(redis锁、mysql悲观锁)](https://blog.csdn.net/weixin_39734609/article/details/120302338)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值