背景:需要批量删除key,但keys命令会存在安全隐患,所以只能想到用scan命令,但是php-redis的scan貌似有bug,遍历一遍后,游标没有重新赋值。但是按照官方文档描述,游标初始为null,第一遍扫描后,游标应该会大于0【在key足够多的情况下】,实际测试的结果是游标还是null,代码如下:

 1 $it = null;
 2 $key=[];
 3 do {
 4       $redis->setOption(\Redis::OPT_SCAN, \Redis::SCAN_RETRY);
 5       $arr_keys = $redis->scan($it, "{$this->_prefix}*",100);
 6        if ($arr_keys !== FALSE) {
 7             foreach($arr_keys as $str_key) {
 8                  $keys[] = $str_key;
 9              }
10         }
11 } while ($it > 0);
12 $keys = array_unique($keys);
13 $redis->del($keys);

 

本以为上面的代码可以完美地解决问题,但是~~~~~~,这是个死循环,每一次扫描结束后,游标值仍为null,很🤷‍♀️

经查看源码发现,$it接收的是一个地址,但是php函数在调用时无法进行引用传参,会报错,所以就想到了函数回调`call_user_func_array`方法,如下:

 1 $it = null;
 2 $keys = [];
 3        
 4 do {
 5    $redis->setOption(\Redis::OPT_SCAN, \Redis::SCAN_RETRY);
 6    $arr_keys = call_user_func_array(array($redis, 'scan'), array(&$it, "{$this->_prefix}*",100));
 7    if ($arr_keys !== FALSE) {
 8        foreach($arr_keys as $str_key) {
 9            $keys[] = $str_key;
10        }
11    }
12 } while ($it > 0);
13 $keys = array_unique($keys);
14 $redis->del($keys);

问题完美解决。

 

PS:php-redis的包是不是有bug呢,欢迎大家一起讨论!!!!!

 

 posted on 2019-09-12 19:04  15mins❤️  阅读(412)  评论(0编辑  收藏  举报