123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997 |
- <?php
- /**
- * redis操作类
- * 说明,任何为false的串,存在redis中都是空串。
- * 只有在key不存在时,才会返回false。
- * 这点可用于防止缓存穿透
- *
- */
- class Redisgo
- {
- private $redis;
-
- //当前数据库ID号
- protected $dbId=0;
-
- //当前权限认证码
- protected $auth;
-
- /**
- * 实例化的对象,单例模式.
- * @var \iphp\db\Redis
- */
- static private $_instance=array();
-
- private $k;
-
- //连接属性数组
- protected $attr=array(
- //连接超时时间,redis配置文件中默认为300秒
- 'timeout'=>30,
- //选择的数据库。
- 'db_id'=>0,
- );
-
- //什么时候重新建立连接
- protected $expireTime;
-
- protected $host;
-
- protected $port;
-
-
- private function __construct($config,$attr=array())
- {
- $this->attr = array_merge($this->attr,$attr);
- $this->redis = new Redis();
- $this->port = $config['port'] ? $config['port'] : 6379;
- $this->host = $config['host'];
- $this->redis->connect($this->host, $this->port, $this->attr['timeout']);
-
- if($config['auth'])
- {
- $this->auth($config['auth']);
- $this->auth = $config['auth'];
- }
-
- $this->expireTime = time() + $this->attr['timeout'];
- }
-
- /**
- * 得到实例化的对象.
- * 为每个数据库建立一个连接
- * 如果连接超时,将会重新建立一个连接
- * @param array $config
- * @param int $dbId
- * @return \iphp\db\Redis
- */
- public static function getInstance($config, $attr = array())
- {
- //如果是一个字符串,将其认为是数据库的ID号。以简化写法。
- if(!is_array($attr))
- {
- $dbId = $attr;
- $attr = array();
- $attr['db_id'] = $dbId;
- }
-
- $attr['db_id'] = $attr['db_id'] ? $attr['db_id'] : 0;
-
-
- $k = md5(implode('', $config).$attr['db_id']);
- if(! (static::$_instance[$k] instanceof self))
- {
-
- static::$_instance[$k] = new self($config,$attr);
- static::$_instance[$k]->k = $k;
- static::$_instance[$k]->dbId = $attr['db_id'];
-
- //如果不是0号库,选择一下数据库。
- if($attr['db_id'] != 0){
- static::$_instance[$k]->select($attr['db_id']);
- }
- }
- elseif( time() > static::$_instance[$k]->expireTime)
- {
- static::$_instance[$k]->close();
- static::$_instance[$k] = new self($config,$attr);
- static::$_instance[$k]->k = $k;
- static::$_instance[$k]->dbId= $attr['db_id'];
-
- //如果不是0号库,选择一下数据库。
- if($attr['db_id']!=0){
- static::$_instance[$k]->select($attr['db_id']);
- }
- }
- return static::$_instance[$k];
- }
-
- private function __clone(){}
-
- /**
- * 执行原生的redis操作
- * @return \Redis
- */
- public function getRedis()
- {
- return $this->redis;
- }
-
- /*****************hash表操作函数*******************/
-
- /**
- * 得到hash表中一个字段的值
- * @param string $key 缓存key
- * @param string $field 字段
- * @return string|false
- */
- public function hGet($key,$field)
- {
- return $this->redis->hGet($key,$field);
- }
-
- /**
- * 为hash表设定一个字段的值
- * @param string $key 缓存key
- * @param string $field 字段
- * @param string $value 值。
- * @return bool
- */
- public function hSet($key,$field,$value)
- {
- return $this->redis->hSet($key,$field,$value);
- }
-
- /**
- * 判断hash表中,指定field是不是存在
- * @param string $key 缓存key
- * @param string $field 字段
- * @return bool
- */
- public function hExists($key,$field)
- {
- return $this->redis->hExists($key,$field);
- }
-
- /**
- * 删除hash表中指定字段 ,支持批量删除
- * @param string $key 缓存key
- * @param string $field 字段
- * @return int
- */
- public function hdel($key,$field)
- {
- $fieldArr=explode(',',$field);
- $delNum=0;
-
- foreach($fieldArr as $row)
- {
- $row=trim($row);
- $delNum+=$this->redis->hDel($key,$row);
- }
-
- return $delNum;
- }
-
- /**
- * 返回hash表元素个数
- * @param string $key 缓存key
- * @return int|bool
- */
- public function hLen($key)
- {
- return $this->redis->hLen($key);
- }
-
- /**
- * 为hash表设定一个字段的值,如果字段存在,返回false
- * @param string $key 缓存key
- * @param string $field 字段
- * @param string $value 值。
- * @return bool
- */
- public function hSetNx($key,$field,$value)
- {
- return $this->redis->hSetNx($key,$field,$value);
- }
-
- /**
- * 为hash表多个字段设定值。
- * @param string $key
- * @param array $value
- * @return array|bool
- */
- public function hMset($key,$value)
- {
- if(!is_array($value))
- return false;
- return $this->redis->hMset($key,$value);
- }
-
- /**
- * 为hash表多个字段设定值。
- * @param string $key
- * @param array|string $value string以','号分隔字段
- * @return array|bool
- */
- public function hMget($key,$field)
- {
- if(!is_array($field))
- $field=explode(',', $field);
- return $this->redis->hMget($key,$field);
- }
-
- /**
- * 为hash表设这累加,可以负数
- * @param string $key
- * @param int $field
- * @param string $value
- * @return bool
- */
- public function hIncrBy($key,$field,$value)
- {
- $value=intval($value);
- return $this->redis->hIncrBy($key,$field,$value);
- }
-
- /**
- * 返回所有hash表的所有字段
- * @param string $key
- * @return array|bool
- */
- public function hKeys($key)
- {
- return $this->redis->hKeys($key);
- }
-
- /**
- * 返回所有hash表的字段值,为一个索引数组
- * @param string $key
- * @return array|bool
- */
- public function hVals($key)
- {
- return $this->redis->hVals($key);
- }
-
- /**
- * 返回所有hash表的字段值,为一个关联数组
- * @param string $key
- * @return array|bool
- */
- public function hGetAll($key)
- {
- return $this->redis->hGetAll($key);
- }
-
- /*********************有序集合操作*********************/
-
- /**
- * 给当前集合添加一个元素
- * 如果value已经存在,会更新order的值。
- * @param string $key
- * @param string $order 序号
- * @param string $value 值
- * @return bool
- */
- public function zAdd($key,$order,$value)
- {
- return $this->redis->zAdd($key,$order,$value);
- }
-
- /**
- * 给$value成员的order值,增加$num,可以为负数
- * @param string $key
- * @param string $num 序号
- * @param string $value 值
- * @return 返回新的order
- */
- public function zinCry($key,$num,$value)
- {
- return $this->redis->zinCry($key,$num,$value);
- }
-
- /**
- * 删除值为value的元素
- * @param string $key
- * @param stirng $value
- * @return bool
- */
- public function zRem($key,$value)
- {
- return $this->redis->zRem($key,$value);
- }
-
- /**
- * 集合以order递增排列后,0表示第一个元素,-1表示最后一个元素
- * @param string $key
- * @param int $start
- * @param int $end
- * @return array|bool
- */
- public function zRange($key,$start,$end)
- {
- return $this->redis->zRange($key,$start,$end);
- }
-
- /**
- * 集合以order递减排列后,0表示第一个元素,-1表示最后一个元素
- * @param string $key
- * @param int $start
- * @param int $end
- * @return array|bool
- */
- public function zRevRange($key,$start,$end)
- {
- return $this->redis->zRevRange($key,$start,$end);
- }
-
- /**
- * 集合以order递增排列后,返回指定order之间的元素。
- * min和max可以是-inf和+inf 表示最大值,最小值
- * @param string $key
- * @param int $start
- * @param int $end
- * @package array $option 参数
- * withscores=>true,表示数组下标为Order值,默认返回索引数组
- * limit=>array(0,1) 表示从0开始,取一条记录。
- * @return array|bool
- */
- public function zRangeByScore($key,$start='-inf',$end="+inf",$option=array())
- {
- return $this->redis->zRangeByScore($key,$start,$end,$option);
- }
-
- /**
- * 集合以order递减排列后,返回指定order之间的元素。
- * min和max可以是-inf和+inf 表示最大值,最小值
- * @param string $key
- * @param int $start
- * @param int $end
- * @package array $option 参数
- * withscores=>true,表示数组下标为Order值,默认返回索引数组
- * limit=>array(0,1) 表示从0开始,取一条记录。
- * @return array|bool
- */
- public function zRevRangeByScore($key,$start='-inf',$end="+inf",$option=array())
- {
- return $this->redis->zRevRangeByScore($key,$start,$end,$option);
- }
-
- /**
- * 返回order值在start end之间的数量
- * @param unknown $key
- * @param unknown $start
- * @param unknown $end
- */
- public function zCount($key,$start,$end)
- {
- return $this->redis->zCount($key,$start,$end);
- }
-
- /**
- * 返回值为value的order值
- * @param unknown $key
- * @param unknown $value
- */
- public function zScore($key,$value)
- {
- return $this->redis->zScore($key,$value);
- }
-
- /**
- * 返回集合以score递增加排序后,指定成员的排序号,从0开始。
- * @param unknown $key
- * @param unknown $value
- */
- public function zRank($key,$value)
- {
- return $this->redis->zRank($key,$value);
- }
-
- /**
- * 返回集合以score递增加排序后,指定成员的排序号,从0开始。
- * @param unknown $key
- * @param unknown $value
- */
- public function zRevRank($key,$value)
- {
- return $this->redis->zRevRank($key,$value);
- }
-
- /**
- * 删除集合中,score值在start end之间的元素 包括start end
- * min和max可以是-inf和+inf 表示最大值,最小值
- * @param unknown $key
- * @param unknown $start
- * @param unknown $end
- * @return 删除成员的数量。
- */
- public function zRemRangeByScore($key,$start,$end)
- {
- return $this->redis->zRemRangeByScore($key,$start,$end);
- }
-
- /**
- * 返回集合元素个数。
- * @param unknown $key
- */
- public function zCard($key)
- {
- return $this->redis->zCard($key);
- }
- /*********************队列操作命令************************/
-
- /**
- * 在队列尾部插入一个元素
- * @param unknown $key
- * @param unknown $value
- * 返回队列长度
- */
- public function rPush($key,$value)
- {
- return $this->redis->rPush($key,$value);
- }
-
- /**
- * 在队列尾部插入一个元素 如果key不存在,什么也不做
- * @param unknown $key
- * @param unknown $value
- * 返回队列长度
- */
- public function rPushx($key,$value)
- {
- return $this->redis->rPushx($key,$value);
- }
-
- /**
- * 在队列头部插入一个元素
- * @param unknown $key
- * @param unknown $value
- * 返回队列长度
- */
- public function lPush($key,$value)
- {
- return $this->redis->lPush($key,$value);
- }
-
- /**
- * 在队列头插入一个元素 如果key不存在,什么也不做
- * @param unknown $key
- * @param unknown $value
- * 返回队列长度
- */
- public function lPushx($key,$value)
- {
- return $this->redis->lPushx($key,$value);
- }
-
- /**
- * 返回队列长度
- * @param unknown $key
- */
- public function lLen($key)
- {
- return $this->redis->lLen($key);
- }
-
- /**
- * 返回队列指定区间的元素
- * @param unknown $key
- * @param unknown $start
- * @param unknown $end
- */
- public function lRange($key,$start,$end)
- {
- return $this->redis->lrange($key,$start,$end);
- }
-
- /**
- * 返回队列中指定索引的元素
- * @param unknown $key
- * @param unknown $index
- */
- public function lIndex($key,$index)
- {
- return $this->redis->lIndex($key,$index);
- }
-
- /**
- * 设定队列中指定index的值。
- * @param unknown $key
- * @param unknown $index
- * @param unknown $value
- */
- public function lSet($key,$index,$value)
- {
- return $this->redis->lSet($key,$index,$value);
- }
-
- /**
- * 删除值为vaule的count个元素
- * PHP-REDIS扩展的数据顺序与命令的顺序不太一样,不知道是不是bug
- * count>0 从尾部开始
- * >0 从头部开始
- * =0 删除全部
- * @param unknown $key
- * @param unknown $count
- * @param unknown $value
- */
- public function lRem($key,$count,$value)
- {
- return $this->redis->lRem($key,$value,$count);
- }
-
- /**
- * 删除并返回队列中的头元素。
- * @param unknown $key
- */
- public function lPop($key)
- {
- return $this->redis->lPop($key);
- }
-
- /**
- * 删除并返回队列中的尾元素
- * @param unknown $key
- */
- public function rPop($key)
- {
- return $this->redis->rPop($key);
- }
-
- /*************redis字符串操作命令*****************/
-
- /**
- * 设置一个key
- * @param unknown $key
- * @param unknown $value
- */
- public function set($key,$value)
- {
- return $this->redis->set($key,$value);
- }
-
- /**
- * 得到一个key
- * @param unknown $key
- */
- public function get($key)
- {
- return $this->redis->get($key);
- }
-
- public function incr($key)
- {
- return $this->redis->incr($key);
- }
-
- //incrBy 如果填写了第二个参数,者自增第二个参数所填的值
- public function incrBy($key,$value)
- {
- return $this->redis->incrBy($key,$value);
- }
-
- public function decr($key)
- {
- return $this->redis->decr($key);
- }
-
- //incrBy 如果填写了第二个参数,者自增第二个参数所填的值
- public function decrBy($key,$value)
- {
- return $this->redis->decrBy($key,$value);
- }
-
-
- /**
- * 设置一个有过期时间的key
- * @param unknown $key
- * @param unknown $expire
- * @param unknown $value
- */
- public function setex($key,$expire,$value)
- {
- return $this->redis->setex($key,$expire,$value);
- }
-
-
- /**
- * 设置一个key,如果key存在,不做任何操作.
- * @param unknown $key
- * @param unknown $value
- */
- public function setnx($key,$value)
- {
- return $this->redis->setnx($key,$value);
- }
-
- /**
- * 批量设置key
- * @param unknown $arr
- */
- public function mset($arr)
- {
- return $this->redis->mset($arr);
- }
-
- /*************redis 无序集合操作命令*****************/
-
- /**
- * 返回集合中所有元素
- * @param unknown $key
- */
- public function sMembers($key)
- {
- return $this->redis->sMembers($key);
- }
-
- /**
- * 求2个集合的差集
- * @param unknown $key1
- * @param unknown $key2
- */
- public function sDiff($key1,$key2)
- {
- return $this->redis->sDiff($key1,$key2);
- }
-
- /**
- * 添加集合。由于版本问题,扩展不支持批量添加。这里做了封装
- * @param unknown $key
- * @param string|array $value
- */
- public function sAdd($key,$value)
- {
- if(!is_array($value))
- $arr=array($value);
- else
- $arr=$value;
- foreach($arr as $row)
- $this->redis->sAdd($key,$row);
- }
-
- /**
- * 返回无序集合的元素个数
- * @param unknown $key
- */
- public function scard($key)
- {
- return $this->redis->scard($key);
- }
-
- /**
- * 从集合中删除一个元素
- * @param unknown $key
- * @param unknown $value
- */
- public function srem($key,$value)
- {
- return $this->redis->srem($key,$value);
- }
-
- /*************redis管理操作命令*****************/
-
- /**
- * 选择数据库
- * @param int $dbId 数据库ID号
- * @return bool
- */
- public function select($dbId)
- {
- $this->dbId=$dbId;
- return $this->redis->select($dbId);
- }
-
- /**
- * 清空当前数据库
- * @return bool
- */
- public function flushDB()
- {
- return $this->redis->flushDB();
- }
-
- /**
- * 返回当前库状态
- * @return array
- */
- public function info()
- {
- return $this->redis->info();
- }
-
- /**
- * 同步保存数据到磁盘
- */
- public function save()
- {
- return $this->redis->save();
- }
-
- /**
- * 异步保存数据到磁盘
- */
- public function bgSave()
- {
- return $this->redis->bgSave();
- }
-
- /**
- * 返回最后保存到磁盘的时间
- */
- public function lastSave()
- {
- return $this->redis->lastSave();
- }
-
- /**
- * 返回key,支持*多个字符,?一个字符
- * 只有* 表示全部
- * @param string $key
- * @return array
- */
- public function keys($key)
- {
- return $this->redis->keys($key);
- }
-
- /**
- * 删除指定key
- * @param unknown $key
- */
- public function del($key)
- {
- return $this->redis->del($key);
- }
-
- /**
- * 判断一个key值是不是存在
- * @param unknown $key
- */
- public function exists($key)
- {
- return $this->redis->exists($key);
- }
-
- /**
- * 为一个key设定过期时间 单位为秒
- * @param unknown $key
- * @param unknown $expire
- */
- public function expire($key,$expire)
- {
- return $this->redis->expire($key,$expire);
- }
-
- /**
- * 返回一个key还有多久过期,单位秒
- * @param unknown $key
- */
- public function ttl($key)
- {
- return $this->redis->ttl($key);
- }
-
- /**
- * 设定一个key什么时候过期,time为一个时间戳
- * @param unknown $key
- * @param unknown $time
- */
- public function exprieAt($key,$time)
- {
- return $this->redis->expireAt($key,$time);
- }
-
- /**
- * 关闭服务器链接
- */
- public function close()
- {
- return $this->redis->close();
- }
-
- /**
- * 关闭所有连接
- */
- public static function closeAll()
- {
- foreach(static::$_instance as $o)
- {
- if($o instanceof self)
- $o->close();
- }
- }
-
- /** 这里不关闭连接,因为session写入会在所有对象销毁之后。
- public function __destruct()
- {
- return $this->redis->close();
- }
- **/
- /**
- * 返回当前数据库key数量
- */
- public function dbSize()
- {
- return $this->redis->dbSize();
- }
-
- /**
- * 返回一个随机key
- */
- public function randomKey()
- {
- return $this->redis->randomKey();
- }
-
- /**
- * 得到当前数据库ID
- * @return int
- */
- public function getDbId()
- {
- return $this->dbId;
- }
-
- /**
- * 返回当前密码
- */
- public function getAuth()
- {
- return $this->auth;
- }
-
- public function getHost()
- {
- return $this->host;
- }
-
- public function getPort()
- {
- return $this->port;
- }
-
- public function getConnInfo()
- {
- return array(
- 'host'=>$this->host,
- 'port'=>$this->port,
- 'auth'=>$this->auth
- );
- }
- /*********************事务的相关方法************************/
-
- /**
- * 监控key,就是一个或多个key添加一个乐观锁
- * 在此期间如果key的值如果发生的改变,刚不能为key设定值
- * 可以重新取得Key的值。
- * @param unknown $key
- */
- public function watch($key)
- {
- return $this->redis->watch($key);
- }
-
- /**
- * 取消当前链接对所有key的watch
- * EXEC 命令或 DISCARD 命令先被执行了的话,那么就不需要再执行 UNWATCH 了
- */
- public function unwatch()
- {
- return $this->redis->unwatch();
- }
-
- /**
- * 开启一个事务
- * 事务的调用有两种模式Redis::MULTI和Redis::PIPELINE,
- * 默认是Redis::MULTI模式,
- * Redis::PIPELINE管道模式速度更快,但没有任何保证原子性有可能造成数据的丢失
- */
- public function multi($type=\Redis::MULTI)
- {
- return $this->redis->multi($type);
- }
-
- /**
- * 执行一个事务
- * 收到 EXEC 命令后进入事务执行,事务中任意命令执行失败,其余的命令依然被执行
- */
- public function exec()
- {
- return $this->redis->exec();
- }
-
- /**
- * 回滚一个事务
- */
- public function discard()
- {
- return $this->redis->discard();
- }
-
- /**
- * 测试当前链接是不是已经失效
- * 没有失效返回+PONG
- * 失效返回false
- */
- public function ping()
- {
- return $this->redis->ping();
- }
-
- public function auth($auth)
- {
- return $this->redis->auth($auth);
- }
- /*********************自定义的方法,用于简化操作************************/
-
- /**
- * 得到一组的ID号
- * @param unknown $prefix
- * @param unknown $ids
- */
- public function hashAll($prefix,$ids)
- {
- if($ids==false)
- return false;
- if(is_string($ids))
- $ids=explode(',', $ids);
- $arr=array();
- foreach($ids as $id)
- {
- $key=$prefix.'.'.$id;
- $res=$this->hGetAll($key);
- if($res!=false)
- $arr[]=$res;
- }
-
- return $arr;
- }
-
- /**
- * 生成一条消息,放在redis数据库中。使用0号库。
- * @param string|array $msg
- */
- public function pushMessage($lkey,$msg)
- {
- if(is_array($msg)){
- $msg = json_encode($msg);
- }
- $key = md5($msg);
-
- //如果消息已经存在,删除旧消息,已当前消息为准
- //echo $n=$this->lRem($lkey, 0, $key)."\n";
- //重新设置新消息
- $this->lPush($lkey, $key);
- $this->setex($key, 3600, $msg);
- return $key;
- }
-
-
- /**
- * 得到条批量删除key的命令
- * @param unknown $keys
- * @param unknown $dbId
- */
- public function delKeys($keys,$dbId)
- {
- $redisInfo=$this->getConnInfo();
- $cmdArr=array(
- 'redis-cli',
- '-a',
- $redisInfo['auth'],
- '-h',
- $redisInfo['host'],
- '-p',
- $redisInfo['port'],
- '-n',
- $dbId,
- );
- $redisStr=implode(' ', $cmdArr);
- $cmd="{$redisStr} KEYS \"{$keys}\" | xargs {$redisStr} del";
- return $cmd;
- }
- }
|