Как удалить элементы с тем же ключом префикса в memcached?

например, у меня есть некоторые кэшированные элементы с тем же префиксом, например

'app_111111', 'app_222222', 'app_333333', ...

могу ли я удалить такие элементы "app_xxxxxx" любыми командами memcached?

4 ответов


Memcached не предлагает эту функциональность из коробки, поэтому вы должны построить ее в себе.

Я решаю это, определяя префикс (или пространство имен) в моем приложении для групп ключей. Любой ключ, который я установил в memcached имеет этот префикс перед ним. Всякий раз, когда я хочу "удалить" материал из Memcached, я просто меняю префикс. И всякий раз, когда я хочу найти ключ в Memcached, я добавляю к нему этот префикс.

в вашем случае вы можете начать с установки приставка к, скажем,MyAppPrefix1, Так что ваши ключи будут храниться как MyAppPrefix1::app_333333, MyAppPrefix1::app_444444.

позже, когда вы хотите "удалить" эти записи, установите приложение для использования MyAppPrefix2. Затем, когда вы пытаетесь получить ключ от Memcached называется app_333333, он будет искать MyAppPrefix2::app_333333 и не найдет его в первый раз, как будто он был удален.


Как насчет этой функции в PHP:

function deletekeysbyindex($prefix) {
    $m = new Memcached();
    $m->addServer('localhost', 11211);
    $keys = $m->getAllKeys();
    foreach ($keys as $index => $key) {
        if (strpos($key,$prefix) !== 0) {
            unset($keys[$index]);
        } else {
            $m->delete($key);
        }
    }
    return $keys;
}

удаляет ключи, начинающиеся с $ prefix, и возвращает список всех удаленных ключей. Я запустил это на ключах 30,000+ только что на общем сервере, и это было довольно быстро - вероятно, менее одной секунды.


мы не можем сделать это только в одном запросе memcache. Мы просто можем сделать это:

public function clearByPrefix($prefixes = array()) {
    $prefixes = array_unique($prefixes);

    $slabs = $this->memcache->getExtendedStats('slabs');
    foreach ($slabs as $serverSlabs) {
        if ($serverSlabs) {
            foreach ($serverSlabs as $slabId => $slabMeta) {
                if (is_int($slabId)) {
                    try {
                        $cacheDump = $this->memcache->getExtendedStats('cachedump', (int) $slabId, 1000);
                    } catch (Exception $e) {
                        continue;
                    }

                    if (is_array($cacheDump)) {
                        foreach ($cacheDump as $dump) {
                            if (is_array($dump)) {
                                foreach ($dump as $key => $value) {

                                    $clearFlag = false;
                                    // Check key has prefix or not
                                    foreach ($prefixes as $prefix) {
                                        $clearFlag = $clearFlag || preg_match('/^' . preg_quote($prefix, '/') . '/', $key);
                                    }
                                    // Clear cache
                                    if ($clearFlag) {
                                        $this->clear($key);
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    }
}

и вызовите эту функцию следующим образом:

        $prefixes = array();

        array_push($prefixes, 'prefix1_');
        array_push($prefixes, 'prefix2_');
        array_push($prefixes, 'prefix3_');

        $this->clearByPrefix($prefixes);

это хак, который работает, хотя и немного медленно. На сервере с 0.6 миллионами ключей это заняло полсекунды.

    $prefix = 'MyApp::Test';
    $len = strlen($prefix);

    $proc = popen('/usr/local/bin/memdump --servers=localhost', 'r');
    while (($key = fgets($proc)) !== false) {
        if (substr_compare($key, $prefix, 0, $len) === 0) {
            $memcached->delete(substr($key, 0, -1));
        }
    }