背景:需要批量删除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呢,欢迎大家一起讨论!!!!!