另辟蹊径-检查memcached中指定的key是否存在
??? 我们都知道,memcached的协议中是没有任何一个协议提供给我们检查某个key是否存在的。但是我们可能会有这样的需求。
?
??? 一些开源的客户端提供了KeyExists()这样的方法,但是其实现原理是调用了memcached的get协议,以此来达到判断某个key是否存在的目的:能取到表示这个key存在,取不到则不存在。这样的效率是比较低的,特别是当数据包比较大的情况下。
?
??? 在不读取memcached的数据的情况下,如何高效的判断某个key是否存在呢?
?
??? 大家都知道memcached的cas(check and set)协议吧,该协议在写memcached时先判断是否有人在你之前更新过数据,如果有更新过,则此次写memcached失败。
?
??? 那我们来看一下cas的语法:
?
??? cas key flag exptime bytes version\r\n
?
??? 其中的version是memcached内部维护的一个版本号,是个uint64_t的值,这个值是调用memcached的gets协议获得的,version从1开始,memcached刚启动的时候,节点数为0,从存储第一个节点开始,version不断累加。因此,version是不可能出现0这个情况,这点很重要,是达到我们的目的的一个关键点。
?
??? 好了,我们已经知道了cas的用法,根据它的特性可以实现我们检查一个key是否存在的目的,聪明的读者可能已经知道该怎么做了吧。
?
??? 比如要检查foo这个key是否存在,可以构造这样的命令
???
??? cas foo 0 0?1 0\r\n
?
??? 这里构造一个字节的数据包,将version字段设置成0,尝试写memcached。
?
??? 如果foo已经存在,因为它的version不可能等于0,所以memcached提示EXISTS
??? 如果foo不存在,memcached提示NOT_FOUND
?
------------------------------------------
??? set foo 0 0 3??????????????? #?先写入
??? bar
??? STORED
??? cas foo 0 0 1 0???????????? # 判断foo是否存在,系统提示EXISTS
??? a
??? EXISTS
??? delete foo???????????????????? # 将foo删除
??? DELETED
??? cas foo 0 0 1 0???????????? # 判断foo是否存在,系统提示NOT_FOUND
??? a
??? NOT_FOUND
------------------------------------------
?
??? 这样,我们就知道foo这个key是否存在了。
1 楼 malphi 2010-06-07 的确另辟蹊径。不过貌似找key的需求非常少