首页 > 开发 > PHP > 正文

PHP实现的memcache环形队列类实例

2024-05-04 23:38:12
字体:
来源:转载
供稿:网友

这篇文章主要介绍了PHP实现的memcache环形队列类,实例分析了基于memcache实现环形队列的方法,涉及memcache缓存及队列的相关技巧,需要的朋友可以参考下

本文实例讲述了PHP实现的memcache环形队列类。分享给大家供大家参考。具体如下:

这里介绍了PHP实现的memcache环形队列类。没咋学过数据结构,因为业务需要,所以只是硬着头皮模拟的! 参考PHP memcache 队列代码。为使队列随时可入可出,且不受int长度越界危险(单链采取Head自增的话不作处理有越界可能),所以索性改写成环形队列。可能还有BUG,忘见谅!

 

 
  1. <?php 
  2. /** 
  3. * PHP memcache 环形队列类 
  4. * 原作者 LKK/lianq.net 
  5. * 修改 FoxHunter 
  6. * 因业务需要只保留的队列中的Pop和Push,修改过期时间为0即永久 
  7. */ 
  8. class MQueue 
  9. public static $client
  10. private $expire//过期时间,秒,1~2592000,即30天内 
  11. private $sleepTime//等待解锁时间,微秒 
  12. private $queueName//队列名称,唯一值 
  13. private $retryNum//尝试次数 
  14. private $MAXNUM//最大队列容量 
  15. private $canRewrite//是否可以覆写开关,满出来的内容从头部开始覆盖重写原来的数据 
  16. private $HEAD//下一步要进入的指针位置 
  17. private $TAIL//下一步要进入的指针位置 
  18. private $LEN//队列现有长度 
  19. const LOCK_KEY = '_Fox_MQ_LOCK_'//锁存储标示 
  20. const LENGTH_KEY = '_Fox_MQ_LENGTH_'//队列现长度存储标示 
  21. const VALU_KEY = '_Fox_MQ_VAL_'//队列键值存储标示 
  22. const HEAD_KEY = '_Fox_MQ_HEAD_'//队列HEAD指针位置标示 
  23. const TAIL_KEY = '_Fox_MQ_TAIL_'//队列TAIL指针位置标示 
  24. /* 
  25. * 构造函数 
  26. * 对于同一个$queueName,实例化时必须保障构造函数的参数值一致,否则pop和push会导队列顺序混乱 
  27. */ 
  28. public function __construct($queueName = ''$maxqueue = 1, $canRewrite = false, $expire = 0, $config = ''
  29. if (emptyempty($config)) { 
  30. self::$client = memcache_pconnect('127.0.0.1', 11211); 
  31. elseif (is_array($config)) { //array('host'=>'127.0.0.1','port'=>'11211') 
  32. self::$client = memcache_pconnect($config['host'], $config['port']); 
  33. elseif (is_string($config)) { //"127.0.0.1:11211" 
  34. $tmp = explode(':'$config); 
  35. $conf['host'] = isset($tmp[0]) ? $tmp[0] : '127.0.0.1'
  36. $conf['port'] = isset($tmp[1]) ? $tmp[1] : '11211'
  37. self::$client = memcache_pconnect($conf['host'], $conf['port']); 
  38. if (!self::$client
  39. return false; 
  40. ignore_user_abort(true); //当客户断开连接,允许继续执行 
  41. set_time_limit(0); //取消脚本执行延时上限 
  42. $this->access = false; 
  43. $this->sleepTime = 1000; 
  44. $expire = (emptyempty($expire)) ? 0 : (int) $expire + 1; 
  45. $this->expire = $expire
  46. $this->queueName = $queueName
  47. $this->retryNum = 20000; 
  48. $this->MAXNUM = $maxqueue != null ? $maxqueue : 1; 
  49. $this->canRewrite = $canRewrite
  50. $this->getHeadAndTail(); 
  51. if (!isset($this->HEAD) || emptyempty($this->HEAD)) 
  52. $this->HEAD = 0; 
  53. if (!isset($this->TAIL) || emptyempty($this->TAIL)) 
  54. $this->TAIL = 0; 
  55. if (!isset($this->LEN) || emptyempty($this->LEN)) 
  56. $this->LEN = 0; 
  57. //获取队列首尾指针信息和长度 
  58. private function getHeadAndTail() 
  59. $this->HEAD = (int) memcache_get(self::$client$this->queueName . self::HEAD_KEY); 
  60. $this->TAIL = (int) memcache_get(self::$client$this->queueName . self::TAIL_KEY); 
  61. $this->LEN = (int) memcache_get(self::$client$this->queueName . self::LENGTH_KEY); 
  62. // 利用memcache_add原子性加锁 
  63. private function lock() 
  64. if ($this->access === false) { 
  65. $i = 0; 
  66. while (!memcache_add(self::$client$this->queueName . self::LOCK_KEY, 1, false, $this->expire)) { 
  67. usleep($this->sleepTime); 
  68. @$i++; 
  69. if ($i > $this->retryNum) { //尝试等待N次 
  70. return false; 
  71. break
  72. return $this->access = true; 
  73. return false; 
  74. //更新头部指针指向,指向下一个位置 
  75. private function incrHead() 
  76. //$this->getHeadAndTail(); //获取最新指针信息 ,由于本方法体均在锁内调用,其锁内已调用了此方法,本行注释 
  77. $this->HEAD++; //头部指针下移 
  78. if ($this->HEAD >= $this->MAXNUM) { 
  79. $this->HEAD = 0; //边界值修正 
  80. $this->LEN--; //Head的移动由Pop触发,所以相当于数量减少 
  81. if ($this->LEN < 0) { 
  82. $this->LEN = 0; //边界值修正 
  83. memcache_set(self::$client$this->queueName . self::HEAD_KEY, $this->HEAD, false, $this->expire); //更新 
  84. memcache_set(self::$client$this->queueName . self::LENGTH_KEY, $this->LEN, false, $this->expire); //更新 
  85. //更新尾部指针指向,指向下一个位置 
  86. private function incrTail() 
  87. //$this->getHeadAndTail(); //获取最新指针信息,由于本方法体均在锁内调用,其锁内已调用了此方法,本行注释 
  88. $this->TAIL++; //尾部指针下移 
  89. if ($this->TAIL >= $this->MAXNUM) { 
  90. $this->TAIL = 0; //边界值修正 
  91. $this->LEN++; //Head的移动由Push触发,所以相当于数量增加 
  92. if ($this->LEN >= $this->MAXNUM) { 
  93. $this->LEN = $this->MAXNUM; //边界值长度修正 
  94. memcache_set(self::$client$this->queueName . self::TAIL_KEY, $this->TAIL, false, $this->expire); //更新 
  95. memcache_set(self::$client$this->queueName . self::LENGTH_KEY, $this->LEN, false, $this->expire); //更新 
  96. // 解锁 
  97. private function unLock() 
  98. memcache_delete(self::$client$this->queueName . self::LOCK_KEY); 
  99. $this->access = false; 
  100. //判断是否满队列 
  101. public function isFull() 
  102. //外部直接调用的时候由于没有锁所以此处的值是个大概值,并不很准确,但是内部调用由于在前面有lock,所以可信 
  103. if ($this->canRewrite) 
  104. return false; 
  105. return $this->LEN == $this->MAXNUM ? true : false; 
  106. //判断是否为空 
  107. public function isEmpty() 
  108. //外部直接调用的时候由于没有锁所以此处的值是个大概值,并不很准确,但是内部调用由于在前面有lock,所以可信 
  109. return $this->LEN == 0 ? true : false; 
  110. public function getLen() 
  111. //外部直接调用的时候由于没有锁所以此处的值是个大概值,并不很准确,但是内部调用由于在前面有lock,所以可信 
  112. return $this->LEN; 
  113. /* 
  114. * push值 
  115. * @param mixed 值 
  116. * @return bool 
  117. */ 
  118. public function push($data = ''
  119. $result = false; 
  120. if (emptyempty($data)) 
  121. return $result
  122. if (!$this->lock()) { 
  123. return $result
  124. $this->getHeadAndTail(); //获取最新指针信息 
  125. if ($this->isFull()) { //只有在非覆写下才有Full概念 
  126. $this->unLock(); 
  127. return false; 
  128. if (memcache_set(self::$client$this->queueName . self::VALU_KEY . $this->TAIL, $data, MEMCACHE_COMPRESSED, $this->expire)) { 
  129. //当推送后,发现尾部和头部重合(此时指针还未移动),且右边仍有未由Head读取的数据,那么移动Head指针,避免尾部指针跨越Head 
  130. if ($this->TAIL == $this->HEAD && $this->LEN >= 1) { 
  131. $this->incrHead(); 
  132. $this->incrTail(); //移动尾部指针 
  133. $result = true; 
  134. $this->unLock(); 
  135. return $result
  136. /* 
  137. * Pop一个值 
  138. * @param [length] int 队列长度 
  139. * @return array 
  140. */ 
  141. public function pop($length = 0) 
  142. if (!is_numeric($length)) 
  143. return false; 
  144. if (!$this->lock()) 
  145. return false; 
  146. $this->getHeadAndTail(); 
  147. if (emptyempty($length)) 
  148. $length = $this->LEN; //默认读取所有 
  149. if ($this->isEmpty()) { 
  150. $this->unLock(); 
  151. return false; 
  152. //获取长度超出队列长度后进行修正 
  153. if ($length > $this->LEN) 
  154. $length = $this->LEN; 
  155. $data = $this->popKeyArray($length); 
  156. $this->unLock(); 
  157. return $data
  158. /* 
  159. * pop某段长度的值 
  160. * @param [length] int 队列长度 
  161. * @return array 
  162. */ 
  163. private function popKeyArray($length
  164. $result = array(); 
  165. if (emptyempty($length)) 
  166. return $result
  167. for ($k = 0; $k < $length$k++) { 
  168. $result[] = @memcache_get(self::$client$this->queueName . self::VALU_KEY . $this->HEAD); 
  169. @memcache_delete(self::$client$this->queueName . self::VALU_KEY . $this->HEAD, 0); 
  170. //当提取值后,发现头部和尾部重合(此时指针还未移动),且右边没有数据,即队列中最后一个数据被完全掏空,此时指针停留在本地不移动,队列长度变为0 
  171. if ($this->TAIL == $this->HEAD && $this->LEN <= 1) { 
  172. $this->LEN = 0; 
  173. memcache_set(self::$client$this->queueName . self::LENGTH_KEY, $this->LEN, false, $this->expire); //更新 
  174. break
  175. else { 
  176. $this->incrHead(); //首尾未重合,或者重合但是仍有未读取出的数据,均移动HEAD指针到下一处待读取位置 
  177. return $result
  178. /* 
  179. * 重置队列 
  180. * * @return NULL 
  181. */ 
  182. private function reset($all = false) 
  183. if ($all) { 
  184. memcache_delete(self::$client$this->queueName . self::HEAD_KEY, 0); 
  185. memcache_delete(self::$client$this->queueName . self::TAIL_KEY, 0); 
  186. memcache_delete(self::$client$this->queueName . self::LENGTH_KEY, 0); 
  187. else { 
  188. $this->HEAD = $this->TAIL = $this->LEN = 0; 
  189. memcache_set(self::$client$this->queueName . self::HEAD_KEY, 0, false, $this->expire); 
  190. memcache_set(self::$client$this->queueName . self::TAIL_KEY, 0, false, $this->expire); 
  191. memcache_set(self::$client$this->queueName . self::LENGTH_KEY, 0, false, $this->expire); 
  192. /* 
  193. * 清除所有memcache缓存数据 
  194. * @return NULL 
  195. */ 
  196. public function memFlush() 
  197. memcache_flush(self::$client); 
  198. public function clear($all = false) 
  199. if (!$this->lock()) 
  200. return false; 
  201. $this->getHeadAndTail(); 
  202. $Head = $this->HEAD; 
  203. $Length = $this->LEN; 
  204. $curr = 0; 
  205. for ($i = 0; $i < $Length$i++) { 
  206. $curr = $this->$Head + $i
  207. if ($curr >= $this->MAXNUM) { 
  208. $this->HEAD = $curr = 0; 
  209. @memcache_delete(self::$client$this->queueName . self::VALU_KEY . $curr, 0); 
  210. $this->unLock(); 
  211. $this->reset($all); 
  212. return true; 

希望本文所述对大家的php程序设计有所帮助。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表