1. 程式人生 > >php使用redis的幾種常見方式和用法

php使用redis的幾種常見方式和用法

query email order lec 循環 簡單的 ews count 調用

一、簡單的字符串緩存

比如針對一些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‘ => ‘[email protected]‘,
        ‘receive‘ => ‘[email protected]‘,
        ‘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>";
});

  

php使用redis的幾種常見方式和用法