TTServer.class.php 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. <?php
  2. namespace KIF\Cache;
  3. use KIF\Exception\ParamsException;
  4. use Memcache as MemcacheResource;
  5. use KIF\Debug\Debug;
  6. use KIF\Verify;
  7. use KIF\Core\Config;
  8. /**
  9. * 对php扩展 Memcache 的包装,用于操作 tokyotyrant。
  10. * tokyotyrant是网络接口,其后的存储由tokyocabinet。该存储里的值都是永久的,不像memcache一样,能够设置过期时间。
  11. * @author gaoxiaogang@gmail.com
  12. *
  13. */
  14. class TTServer {
  15. /**
  16. * 存放已建立的 Memcache 实例
  17. * @var array
  18. */
  19. static private $objMemcaches = array();
  20. /**
  21. *
  22. * 当前所使用的服务器集群
  23. * @var array
  24. */
  25. private $servers;
  26. /**
  27. *
  28. * 保存当前实例下到Memcached的连接
  29. * @var Memcache
  30. */
  31. private $objMemcache;
  32. public function __construct($cluster_flag = 'default') {
  33. $ttserverConfig = Config::getInstance()->get('ttserver');
  34. if (!$ttserverConfig || !isset($ttserverConfig[$cluster_flag])) {
  35. throw new ParamsException("不存在的ttserver集群标志 {$cluster_flag}");
  36. }
  37. $this->servers = $ttserverConfig[$cluster_flag];
  38. if (!isset(self::$objMemcaches[$cluster_flag])) {
  39. $objMemcache = new MemcacheResource();
  40. foreach ($this->servers as $server) {
  41. $tmpHost = isset($server['host']) ? $server['host'] : '127.0.0.1';
  42. $tmpPort = isset($server['port']) ? $server['port'] : '11211';
  43. $tmpPersistent = isset($server['persistent']) ? (bool) $server['persistent'] : false;
  44. $tmpWeight = isset($server['weight']) ? $server['weight'] : 1;
  45. $tmpTimeout = isset($server['timeout']) ? $server['timeout'] : 2;// 默认2秒的超时
  46. $tmpRetry_interval = isset($server['retry_interval']) ? $server['retry_interval'] : 5;// 默认5秒超时重试
  47. $tmpResult = $objMemcache->addServer($tmpHost, $tmpPort, $tmpPersistent
  48. , $tmpWeight, $tmpTimeout, $tmpRetry_interval
  49. );
  50. }
  51. self::$objMemcaches[$cluster_flag] = $objMemcache;
  52. }
  53. $this->objMemcache = self::$objMemcaches[$cluster_flag];
  54. }
  55. /**
  56. *
  57. * 给 $key 设置对应的值
  58. * @param string $key
  59. * @param mixed $value
  60. * @return Boolean 成功:true;失败:false
  61. *
  62. */
  63. public function set($key, $value) {
  64. $begin_microtime = Debug::getTime();
  65. if (!is_scalar($value)) {
  66. $value = serialize($value);
  67. }
  68. $tmpResult = $this->objMemcache->set($key, $value);
  69. Debug::cache($this->servers, $key, Debug::getTime() - $begin_microtime, $tmpResult, 'set');
  70. return $tmpResult;
  71. }
  72. /**
  73. *
  74. * 获取 $key 对应的值
  75. * @param string $key
  76. * @return false | mixed 存在:返回值;值不存在 或 其它原因导致的获取不到值:返回 false。
  77. *
  78. */
  79. public function get($key) {
  80. $begin_microtime = Debug::getTime();
  81. $tmpResult = $this->objMemcache->get($key);
  82. if ($tmpResult !== false) {
  83. # 尝试序列化
  84. $tmpUnserializeVal = @unserialize($tmpResult);
  85. if ($tmpUnserializeVal !== false) {// 反序列化成功,则使用序列化后的值
  86. $tmpResult = $tmpUnserializeVal;
  87. }
  88. Debug::cache($this->servers, $key, Debug::getTime() - $begin_microtime, $tmpResult, 'get');
  89. return $tmpResult;
  90. }
  91. Debug::cache($this->servers, $key, Debug::getTime() - $begin_microtime, false, 'get');
  92. return false;
  93. }
  94. /**
  95. *
  96. * 批量获取 $keys 对应的值
  97. * @param array $keys
  98. * @return false | array 失败:返回false,获取其中任何一个key出错,都会返回false;
  99. * 成功:array,值不存在的key,对应的值是null。
  100. */
  101. public function gets(array $keys) {
  102. if (empty($keys)) {
  103. return array();
  104. }
  105. $begin_microtime = Debug::getTime();
  106. $tmpResult = $this->objMemcache->get($keys);
  107. Debug::cache($this->servers, print_r($keys, true), Debug::getTime() - $begin_microtime, false, 'gets');
  108. if ($tmpResult === false) {
  109. return false;
  110. }
  111. $return = array();
  112. foreach ($keys as $key) {
  113. if (isset($tmpResult[$key])) {
  114. # 尝试序列化
  115. $tmpUnserializeVal = @unserialize($tmpResult[$key]);
  116. if ($tmpUnserializeVal !== false) {// 反序列化成功,则使用序列化后的值
  117. $return[$key] = $tmpUnserializeVal;
  118. } else {
  119. $return[$key] = $tmpResult[$key];
  120. }
  121. } else {
  122. $return[$key] = null;
  123. }
  124. }
  125. return $return;
  126. }
  127. /**
  128. *
  129. * 删除一个元素
  130. * @param string $key 要删除的key
  131. * @return null | boolean 成功:true;失败:false;
  132. */
  133. public function delete($key) {
  134. $begin_microtime = Debug::getTime();
  135. $tmpResult = $this->objMemcache->delete($key);
  136. if ($tmpResult) {
  137. Debug::cache($this->servers, $key, Debug::getTime() - $begin_microtime, true, 'delete');
  138. return true;
  139. }
  140. Debug::cache($this->servers, $key, Debug::getTime() - $begin_microtime, false, 'delete');
  141. return false;
  142. }
  143. /**
  144. *
  145. * 向一个新的key下面增加一个元素
  146. * @param string $key 用于存储值的键名。
  147. * @param mixed $value 存储的值
  148. * @return boolean 成功:true;失败:false;
  149. */
  150. public function add($key, $value) {
  151. $begin_microtime = Debug::getTime();
  152. if (!is_scalar($value)) {
  153. $value = serialize($value);
  154. }
  155. $tmpResult = $this->objMemcache->add($key, $value);
  156. if ($tmpResult) {
  157. Debug::cache($this->servers, "{$key}::{$value}", Debug::getTime() - $begin_microtime, true, 'add');
  158. return true;
  159. }
  160. Debug::cache($this->servers, "{$key}::{$value}", Debug::getTime() - $begin_microtime, false, 'add');
  161. return false;
  162. }
  163. /**
  164. *
  165. * 替换已存在key下的元素
  166. * 如果 服务端不存在key,操作将失败。
  167. * @param string $key 用于存储值的键名。
  168. * @param mixed $value 存储的值
  169. * @return null | boolean 成功:true;失败:false;
  170. */
  171. public function replace($key, $value) {
  172. $begin_microtime = Debug::getTime();
  173. if (!is_scalar($value)) {
  174. $value = serialize($value);
  175. }
  176. $tmpResult = $this->objMemcache->replace($key, $value);
  177. if ($tmpResult) {
  178. Debug::cache($this->servers, "{$key}::{$value}", Debug::getTime() - $begin_microtime, true, 'replace');
  179. return true;
  180. }
  181. Debug::cache($this->servers, "{$key}::{$value}", Debug::getTime() - $begin_microtime, false, 'replace');
  182. return false;
  183. }
  184. /**
  185. *
  186. * 将一个数值元素增加参数offset指定的大小。如果元素的值不是数值类型,返回false
  187. * 经实验发现,Memcached 扩展的increment方法存在bug,这里对其做了修复。
  188. * @param string $key 要增加值的元素的key。
  189. * @param int $offset 要将元素的值增加的大小。
  190. * @return int | false 成功:int(递增后的值);失败或key不存在:false
  191. * !!! 如果 $key 对应的值不是 int 型:通过 $this->get 方法返回的值不是 int 型
  192. * a、如果不是 boolean 型:结果是0
  193. * b、如果是 boolean 型:会以0为基准递增值。
  194. * !!! 所以,请尽量尽量不要对非 int 型的值使用 increment 方法
  195. */
  196. public function increment($key, $offset = 1) {
  197. $begin_microtime = Debug::getTime();
  198. $tmpGetResult = $this->get($key);
  199. if (is_null($tmpGetResult)) {// $key 对应的值不存在
  200. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, null, 'increment');
  201. return false;
  202. }
  203. if (!Verify::naturalNumber($tmpGetResult)) {
  204. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, false, 'increment');
  205. return false;
  206. }
  207. $tmpResult = $this->objMemcache->increment($key, $offset);
  208. if (Verify::unsignedInt($tmpResult)) {
  209. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, $tmpResult, 'increment');
  210. return $tmpResult;
  211. }
  212. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, false, 'increment');
  213. return false;
  214. }
  215. /**
  216. *
  217. * 减小一个数值元素的值,减小多少由参数offset决定。如果元素的值不是数值类型,返回false。
  218. * 如果减小后的值小于0,则新的值被设置为0。
  219. * 如果元素不存在,返回 null。
  220. *
  221. * 经实验发现,Memcached 扩展的decrement方法存在bug,这里对其做了修复。
  222. * @param string $key 要减少值的元素的key。
  223. * @param int $offset 要将元素的值减少的大小。
  224. * @return int | false 成功:int(减少后的值,有可能是 0);失败或key不存在:false;
  225. */
  226. public function decrement($key, $offset = 1) {
  227. $begin_microtime = Debug::getTime();
  228. $tmpGetResult = $this->get($key);
  229. if (is_null($tmpGetResult)) {// $key 对应的值不存在
  230. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, null, 'decrement');
  231. return false;
  232. }
  233. if (!Verify::naturalNumber($tmpGetResult)) {
  234. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, false, 'decrement');
  235. return false;
  236. }
  237. $tmpResult = $this->objMemcache->decrement($key, $offset);
  238. if (Verify::naturalNumber($tmpResult)) {
  239. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, $tmpResult, 'decrement');
  240. return $tmpResult;
  241. }
  242. Debug::cache($this->servers, "{$key}::{$offset}", Debug::getTime() - $begin_microtime, false, 'decrement');
  243. return false;
  244. }
  245. }