亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > 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程序設計有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久久久中文字幕2018| 日韩中文字幕第一页| 成人久久久久久久| 欧美日韩一区二区三区| 上原亚衣av一区二区三区| 欧美成年人视频网站| 亚洲欧美日韩另类| 日本成人激情视频| 精品久久香蕉国产线看观看亚洲| 欧美在线视频网站| 欧美成人精品h版在线观看| 国模gogo一区二区大胆私拍| 国产成人精品电影久久久| 久久久999国产精品| 欧美视频专区一二在线观看| 欧美日本中文字幕| 国产精品美女久久久久av超清| 欧美日韩国产一区在线| 亚洲成人在线网| 欧美日韩一区二区在线| 久热精品视频在线观看| 日韩电影在线观看中文字幕| 国产亚洲成精品久久| 欧美综合激情网| 日韩成人激情影院| 日韩亚洲在线观看| 国产欧美日韩精品在线观看| 亚洲a∨日韩av高清在线观看| 欧美亚洲视频在线看网址| 在线观看日韩欧美| 亚洲一区二区三区xxx视频| 欧美孕妇毛茸茸xxxx| 国产欧美亚洲精品| 亚洲伊人成综合成人网| 成人网在线视频| 久久精品一偷一偷国产| 日韩专区在线观看| 久久精品色欧美aⅴ一区二区| 6080yy精品一区二区三区| 亚洲一区二区三区sesese| 亚洲国产91色在线| 国内免费精品永久在线视频| 欧美亚洲另类激情另类| 国产精品久久久久久久久久三级| 日韩高清av一区二区三区| 成人免费看黄网站| 97视频在线观看免费| 亚洲日本欧美日韩高观看| 国内揄拍国内精品少妇国语| 国产成人一区二区三区电影| 性亚洲最疯狂xxxx高清| 福利一区视频在线观看| 欧美麻豆久久久久久中文| 国产ts一区二区| 亚洲精品国产suv| 国产精品自产拍在线观看中文| 亚洲一区二区三区在线免费观看| 精品亚洲一区二区三区在线播放| 欧美放荡办公室videos4k| 日韩精品中文字幕在线观看| 国产最新精品视频| 精品久久久久久久久久ntr影视| 亚洲国产第一页| 中文字幕精品一区二区精品| 日韩电影中文字幕| 国产日韩在线一区| 一区二区成人精品| 国产热re99久久6国产精品| 国产精品99久久久久久久久久久久| 亚洲欧美在线免费观看| 伊人av综合网| 中文字幕自拍vr一区二区三区| 欧美精品videosex牲欧美| 国产亚洲精品美女| 久久久精品视频在线观看| 国产精品www| 国内精品模特av私拍在线观看| 大荫蒂欧美视频另类xxxx| 亚洲片国产一区一级在线观看| 国产精品丝袜白浆摸在线| 久久精品久久久久| 精品久久久久久久久中文字幕| 欧美xxxx14xxxxx性爽| 亚洲日本中文字幕免费在线不卡| 国产在线观看91精品一区| 国产日韩在线播放| 久久亚洲国产精品成人av秋霞| 性欧美激情精品| 国产日韩综合一区二区性色av| 国产精品青青在线观看爽香蕉| 伊人久久五月天| 国产不卡一区二区在线播放| 色一情一乱一区二区| 久久久久久久999精品视频| 2020国产精品视频| 午夜精品福利视频| 成人福利免费观看| 日韩电影中文 亚洲精品乱码| 国产成人精品综合久久久| 成人精品视频在线| 欧美日韩激情视频8区| 日韩黄色高清视频| 亚洲成人三级在线| 夜夜躁日日躁狠狠久久88av| 91精品国产综合久久香蕉| 亚洲精品一区在线观看香蕉| 日韩av在线直播| 亚洲欧美中文日韩v在线观看| 国产精品扒开腿做| 欧美激情在线有限公司| 欧美精品18videos性欧美| 久久久久久久久久婷婷| 精品久久久国产| 欧美一乱一性一交一视频| 欧美一性一乱一交一视频| 97在线视频国产| 久久久黄色av| 中文字幕成人精品久久不卡| 久久久亚洲成人| 欧美一级片免费在线| 96国产粉嫩美女| 一区二区三区日韩在线| 亚洲精品日韩欧美| 上原亚衣av一区二区三区| 精品成人乱色一区二区| 久久久久免费精品国产| 最新国产成人av网站网址麻豆| 亚洲国产91色在线| 亚洲精品videossex少妇| 日韩欧美成人精品| 亚洲一区二区三区在线免费观看| 国产成人在线精品| 亚洲 日韩 国产第一| 亚洲成色777777在线观看影院| 亚洲精品午夜精品| 亚洲人成亚洲人成在线观看| 国产精品欧美日韩一区二区| 精品一区二区三区三区| 亚洲国产精品久久久久久| 欧美巨乳在线观看| 亚洲bt欧美bt日本bt| 亚洲精品久久久久中文字幕二区| 国产视频综合在线| 久久免费精品视频| 国产欧美精品一区二区| 成人激情电影一区二区| 国产成人av网址| 亚洲一级一级97网| 欧美激情乱人伦一区| 国产精品扒开腿做爽爽爽视频| 久久久国产精品视频| 欧美日本亚洲视频| 欧美精品videosex极品1| 国产色视频一区| 日韩中文字幕在线播放| 亚洲第一精品夜夜躁人人躁| 亚洲精品视频免费在线观看| 一区二区在线视频| 中文字幕亚洲国产| 国产成人精品一区二区| 国产一区深夜福利| 亚洲永久免费观看| 亚洲欧洲高清在线| 亚洲欧美国产高清va在线播|