Как удалить элементы с тем же ключом префикса в 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));
}
}