php使用redis最常用示例 php使用redis的几种常见方式和用法

一、简单的字符串缓存

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<?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脚本从队列中读取任务去发送邮件。

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<?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。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<?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 来实现悲观锁

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
<?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 完成发布和订阅

发布代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<?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);
}

订阅代码:

1
2
3
4
5
6
7
8
9
10
<?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>" ;
});

一、简单的字符串缓存

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<?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脚本从队列中读取任务去发送邮件。

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<?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。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<?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 来实现悲观锁

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
<?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 完成发布和订阅

发布代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<?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);
}

订阅代码:

1
2
3
4
5
6
7
8
9
10
<?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>" ;
});

猜你喜欢

转载自www.cnblogs.com/wzg-bky/p/11357225.html