php的memcache类分享(memcache队列)

2015-01-24信息快讯网

这篇文章主要介绍了php的memcache类的使用方法(memcache队列),需要的朋友可以参考下

memcacheQueue.class.php

<?php
/**
 * PHP memcache 队列类
 * @author LKK/lianq.net
 * @version 0.3
 * @修改说明:
 * 1.放弃了之前的AB面轮值思路,使用类似数组的构造,重写了此类.
 * 2.队列默认先进先出,但增加了反向读取功能.
 * 3.感谢网友FoxHunter提出的宝贵意见.
 * @example:
 * $obj = new memcacheQueue('duilie');
 * $obj->add('1asdf');
 * $obj->getQueueLength();
 * $obj->read(10);
 * $obj->get(8);
 */
class memcacheQueue{
 public static $client;   //memcache客户端连接
 public   $access;   //队列是否可更新
 private   $expire;   //过期时间,秒,1~2592000,即30天内
 private   $sleepTime;   //等待解锁时间,微秒
 private   $queueName;   //队列名称,唯一值
 private   $retryNum;   //重试次数,= 10 * 理论并发数
 public   $currentHead;  //当前队首值
 public   $currentTail;  //当前队尾值

const MAXNUM = 20000; //最大队列数,建议上限10K const HEAD_KEY = '_lkkQueueHead_'; //队列首kye const TAIL_KEY = '_lkkQueueTail_'; //队列尾key const VALU_KEY = '_lkkQueueValu_'; //队列值key const LOCK_KEY = '_lkkQueueLock_'; //队列锁key

/** * 构造函数 * @param string $queueName 队列名称 * @param int $expire 过期时间 * @param array $config memcache配置 * * @return <type> */ public function __construct($queueName ='',$expire=0,$config =''){ if(empty($config)){ self::$client = memcache_pconnect('127.0.0.1',11211); }elseif(is_array($config)){//array('host'=>'127.0.0.1','port'=>'11211') self::$client = memcache_pconnect($config['host'],$config['port']); }elseif(is_string($config)){//"127.0.0.1:11211" $tmp = explode(':',$config); $conf['host'] = isset($tmp[0]) ? $tmp[0] : '127.0.0.1'; $conf['port'] = isset($tmp[1]) ? $tmp[1] : '11211'; self::$client = memcache_pconnect($conf['host'],$conf['port']); } if(!self::$client) return false; ignore_user_abort(true);//当客户断开连接,允许继续执行 set_time_limit(0);//取消脚本执行延时上限 $this->access = false; $this->sleepTime = 1000; $expire = empty($expire) ? 3600 : intval($expire)+1; $this->expire = $expire; $this->queueName = $queueName; $this->retryNum = 1000; $this->head_key = $this->queueName . self::HEAD_KEY; $this->tail_key = $this->queueName . self::TAIL_KEY; $this->lock_key = $this->queueName . self::LOCK_KEY; $this->_initSetHeadNTail(); } /** * 初始化设置队列首尾值 */ private function _initSetHeadNTail(){ //当前队列首的数值 $this->currentHead = memcache_get(self::$client, $this->head_key); if($this->currentHead === false) $this->currentHead =0; //当前队列尾的数值 $this->currentTail = memcache_get(self::$client, $this->tail_key); if($this->currentTail === false) $this->currentTail =0; } /** * 当取出元素时,改变队列首的数值 * @param int $step 步长值 */ private function _changeHead($step=1){ $this->currentHead += $step; memcache_set(self::$client, $this->head_key,$this->currentHead,false,$this->expire); } /** * 当添加元素时,改变队列尾的数值 * @param int $step 步长值 * @param bool $reverse 是否反向 * @return null */ private function _changeTail($step=1, $reverse =false){ if(!$reverse){ $this->currentTail += $step; }else{ $this->currentTail -= $step; } memcache_set(self::$client, $this->tail_key,$this->currentTail,false,$this->expire); } /** * 队列是否为空 * @return bool */ private function _isEmpty(){ return (bool)($this->currentHead === $this->currentTail); } /** * 队列是否已满 * @return bool */ private function _isFull(){ $len = $this->currentTail - $this->currentHead; return (bool)($len === self::MAXNUM); } /** * 队列加锁 */ private function _getLock(){ if($this->access === false){ while(!memcache_add(self::$client, $this->lock_key, 1, false, $this->expire) ){ usleep($this->sleepTime); @$i++; if($i > $this->retryNum){//尝试等待N次 return false; break; } } $this->_initSetHeadNTail(); return $this->access = true; } return $this->access; } /** * 队列解锁 */ private function _unLock(){ memcache_delete(self::$client, $this->lock_key, 0); $this->access = false; } /** * 获取当前队列的长度 * 该长度为理论长度,某些元素由于过期失效而丢失,真实长度<=该长度 * @return int */ public function getQueueLength(){ $this->_initSetHeadNTail(); return intval($this->currentTail - $this->currentHead); } /** * 添加队列数据 * @param void $data 要添加的数据 * @return bool */ public function add($data){ if(!$this->_getLock()) return false; if($this->_isFull()){ $this->_unLock(); return false; } $value_key = $this->queueName . self::VALU_KEY . strval($this->currentTail +1); $result = memcache_set(self::$client, $value_key, $data, MEMCACHE_COMPRESSED, $this->expire); if($result){ $this->_changeTail(); } $this->_unLock(); return $result; } /** * 读取队列数据 * @param int $length 要读取的长度(反向读取使用负数) * @return array */ public function read($length=0){ if(!is_numeric($length)) return false; $this->_initSetHeadNTail(); if($this->_isEmpty()){ return false; } if(empty($length)) $length = self::MAXNUM;//默认所有 $keyArr = array(); if($length >0){//正向读取(从队列首向队列尾) $tmpMin = $this->currentHead; $tmpMax = $tmpMin + $length; for($i= $tmpMin; $i<=$tmpMax; $i++){ $keyArr[] = $this->queueName . self::VALU_KEY . $i; } }else{//反向读取(从队列尾向队列首) $tmpMax = $this->currentTail; $tmpMin = $tmpMax + $length; for($i= $tmpMax; $i >$tmpMin; $i--){ $keyArr[] = $this->queueName . self::VALU_KEY . $i; } } $result = @memcache_get(self::$client, $keyArr); return $result; } /** * 取出队列数据 * @param int $length 要取出的长度(反向读取使用负数) * @return array */ public function get($length=0){ if(!is_numeric($length)) return false; if(!$this->_getLock()) return false; if($this->_isEmpty()){ $this->_unLock(); return false; } if(empty($length)) $length = self::MAXNUM;//默认所有 $length = intval($length); $keyArr = array(); if($length >0){//正向读取(从队列首向队列尾) $tmpMin = $this->currentHead; $tmpMax = $tmpMin + $length; for($i= $tmpMin; $i<=$tmpMax; $i++){ $keyArr[] = $this->queueName . self::VALU_KEY . $i; } $this->_changeHead($length); }else{//反向读取(从队列尾向队列首) $tmpMax = $this->currentTail; $tmpMin = $tmpMax + $length; for($i= $tmpMax; $i >$tmpMin; $i--){ $keyArr[] = $this->queueName . self::VALU_KEY . $i; } $this->_changeTail(abs($length), true); } $result = @memcache_get(self::$client, $keyArr); foreach($keyArr as $v){//取出之后删除 @memcache_delete(self::$client, $v, 0); } $this->_unLock(); return $result; } /** * 清空队列 */ public function clear(){ if(!$this->_getLock()) return false; if($this->_isEmpty()){ $this->_unLock(); return false; } $tmpMin = $this->currentHead--; $tmpMax = $this->currentTail++;

for($i= $tmpMin; $i<=$tmpMax; $i++){ $tmpKey = $this->queueName . self::VALU_KEY . $i; @memcache_delete(self::$client, $tmpKey, 0); } $this->currentTail = $this->currentHead = 0; memcache_set(self::$client, $this->head_key,$this->currentHead,false,$this->expire); memcache_set(self::$client, $this->tail_key,$this->currentTail,false,$this->expire); $this->_unLock(); } /* * 清除所有memcache缓存数据 */ public function memFlush(){ memcache_flush(self::$client); }

}//end class

ThinkPHP后台首页index使用frameset时的注意事项分析
PHP date()函数警告: It is not safe to rely on the system解决方法
PHP中exec函数和shell_exec函数的区别
Linux下PHP连接Oracle数据库
Win7 64位系统下PHP连接Oracle数据库
PHP框架Swoole定时器Timer特性分析
php中用memcached实现页面防刷新功能
zend framework框架中url大小写问题解决方法
PHP命名空间(namespace)的动态访问及使用技巧
PHP命名空间(namespace)的使用基础及示例
php中操作memcached缓存进行增删改查数据的实现代码
win7 64位系统 配置php最新版开发环境(php+Apache+mysql)
linux下安装php的memcached客户端
linux下编译安装memcached服务
php去除换行(回车换行)的三种方法
codeigniter自带数据库类使用方法说明
PHP中nowdoc和heredoc使用需要注意的一点
zf框架的zend_cache缓存使用方法(zend框架)
CodeIgniter框架中_remap()使用方法2例
PHP empty函数报错解决办法
thinkphp路由规则使用示例详解和伪静态功能实现(apache重写)
php检测iis环境是否支持htaccess的方法
jQuery中的RadioButton,input,CheckBox取值赋值实现代码
php将session放入memcached的设置方法
php中的filesystem文件系统函数介绍及使用示例
显示youtube视频缩略图和Vimeo视频缩略图代码分享
PHP下获取上个月、下个月、本月的日期(strtotime,date)
网页上facebook分享功能具体实现
PHP 利用Mail_MimeDecode类提取邮件信息示例
那些年我们错过的魔术方法(Magic Methods)
memcache命令启动参数中文解释
©2014-2024 dbsqp.com