如何使用php查询memcached以获取存储中所有密钥的列表?

时间:2012-03-22 22:19:33

标签: php memcached

我需要将memcached复制到另一个键值系统(couchbase)。如何查询memcached服务器的内容以获取其中的内容列表,以便我可以复制它?

5 个答案:

答案 0 :(得分:13)

Memcache的所有这些解决方案都在 Memcached

function getMemcachedKeys($host = '127.0.0.1', $port = 11211)
{

    $mem = @fsockopen($host, $port);
    if ($mem === FALSE) return -1;

    // retrieve distinct slab
    $r = @fwrite($mem, 'stats items' . chr(10));
    if ($r === FALSE) return -2;

    $slab = array();
    while (($l = @fgets($mem, 1024)) !== FALSE) {
        // sortie ?
        $l = trim($l);
        if ($l == 'END') break;

        $m = array();
        // <STAT items:22:evicted_nonzero 0>
        $r = preg_match('/^STAT\sitems\:(\d+)\:/', $l, $m);
        if ($r != 1) return -3;
        $a_slab = $m[1];

        if (!array_key_exists($a_slab, $slab)) $slab[$a_slab] = array();
    }

    // recuperer les items
    reset($slab);
    foreach ($slab AS $a_slab_key => &$a_slab) {
        $r = @fwrite($mem, 'stats cachedump ' . $a_slab_key . ' 100' . chr(10));
        if ($r === FALSE) return -4;

        while (($l = @fgets($mem, 1024)) !== FALSE) {
            // sortie ?
            $l = trim($l);
            if ($l == 'END') break;

            $m = array();
            // ITEM 42 [118 b; 1354717302 s]
            $r = preg_match('/^ITEM\s([^\s]+)\s/', $l, $m);
            if ($r != 1) return -5;
            $a_key = $m[1];

            $a_slab[] = $a_key;
        }
    }

    // close
    @fclose($mem);
    unset($mem);

    // transform it;
    $keys = array();
    reset($slab);
    foreach ($slab AS &$a_slab) {
        reset($a_slab);
        foreach ($a_slab AS &$a_key) $keys[] = $a_key;
    }
    unset($slab);

    return $keys;
}

答案 1 :(得分:12)

memcache&gt; = 2.0.0

function getMemcacheKeys() {
    $memcache = new Memcache;
    $memcache->connect('127.0.0.1', 11211) 
       or die ("Could not connect to memcache server");

    $list = array();
    $allSlabs = $memcache->getExtendedStats('slabs');
    $items = $memcache->getExtendedStats('items');
    foreach($allSlabs as $server => $slabs) {
        foreach($slabs AS $slabId => $slabMeta) {
            $cdump = $memcache->getExtendedStats('cachedump',(int)$slabId);
            foreach($cdump AS $keys => $arrVal) {
                if (!is_array($arrVal)) continue;
                foreach($arrVal AS $k => $v) {                   
                    echo $k .'<br>';
                }
            }
        }
    }   
} 

答案 2 :(得分:1)

  1. 根据报告https://www.php.net/manual/en/memcached.getallkeys.php#123793函数Memcached::getAllKeys的功能自内存缓存1.4.23开始生效。同样,也不保证所有密钥都可以返回。
  2. @ mark-baker,@ maduka-jayalath和其他基于stats cachedump提供的代码具有局限性,如果您在内存缓存中有数百万个密钥,则不会返回所有密钥。它还对内存缓存服务器的性能有影响。

有一个库可以异步在所有个内存缓存键之间进行迭代,而不会影响性能:https://github.com/qmegas/memcache-search 甚至不需要安装任何内存缓存扩展。

答案 3 :(得分:0)

感谢您提供示例代码

以下是删除特定键或多个键的方法

我正在使用帮助程序类删除缓存,因此您必须为该函数提供对memcache连接的引用

public static function removePriceCache(&$memcache, &$cacheAvailable) {

    if ($cacheAvailable == true) {
        $list = array();
        $allSlabs = $memcache->getExtendedStats('slabs');
        $items = $memcache->getExtendedStats('items');
        foreach ($allSlabs as $server => $slabs) {
            foreach ($slabs AS $slabId => $slabMeta) {
                $cdump = $memcache->getExtendedStats('cachedump', (int) $slabId);
                foreach ($cdump AS $keys => $arrVal) {
                    if (!is_array($arrVal))
                        continue;
                    foreach ($arrVal as $k => $v) {
                        //echo $k . '<br>';

                        if (stripos($k, "Price") !== false) {
                            $memcache->delete($k);
                        }
                    }
                }
            }
        }
    }
}

这将删除所有包含“Price”字样的键。

答案 4 :(得分:0)

根据文档:
https://www.php.net/manual/en/memcached.getallkeys.php

PECL内存缓存> = 2.0.0

$memcached->getAllKeys();