1. LRU アルゴリズムを使用するように Redis を設定します。
LRU (最も最近使用されていない) アルゴリズムは、数多くあるアルゴリズムの 1 つです。置換アルゴリズムの一種。
Redis には maxmemory の概念があり、主に使用されるメモリを固定サイズに制限します。 Redis で使用される LRU アルゴリズムは近似 LRU アルゴリズムです。
(1) maxmemory の設定
上記の通り、maxmemory は Redis の最大メモリ使用量を制限するためのものです。サイズを設定するにはいくつかの方法があります。 1 つの方法は、次のように CONFIG SET を通じて設定することです:
127.0.0.1:6379> CONFIG GET maxmemory 1) "maxmemory" 2) "0" 127.0.0.1:6379> CONFIG SET maxmemory 100MB OK 127.0.0.1:6379> CONFIG GET maxmemory 1) "maxmemory" 2) "104857600"
もう 1 つの方法は、構成ファイル redis.conf を変更することです:
maxmemory 100mb
64 ビット システムでは、maxmemory が次のように設定されることに注意してください。 0 Redis のメモリ使用量が制限されていないことを示します。32 ビット システムでは、maxmemory は暗黙的に 3GB を超えることができません。
Redis のメモリ使用量が指定された制限に達した場合は、交換戦略を選択する必要があります。
(2) 置換ポリシー
Redis のメモリ使用量が maxmemory に達した場合、設定された maxmemory-policy を選択して古いデータを置き換える必要があります。
選択できる置換方法は次のとおりです:
noeviction: 置換なし、つまりメモリが上限に達しても置換は実行されません。メモリ増加を引き起こす可能性のあるコマンドはエラーを返します
allkeys-lru: 新しいデータを保存するために、最近使用頻度が最も低いキーを優先的に削除します
volatile-lru: 新しいデータを保存するために削除する期限切れのキーの中で最も使用頻度の低いキーを選択するのみを設定します
allkeys-random: すべてのキーからいくつかのキーをランダムに選択します。新しいデータを保存するには
次のような置換戦略がどのように実行されるかを理解することが非常に重要です。
#クライアントが新しいコマンドを実行すると、データベースはデータを追加する必要があります (キー値の設定など)(4) 近似 LRU アルゴリズム
Redis の LRU は、厳密な LRU アルゴリズム実装ではなく、主にメモリ占有量の節約とパフォーマンスの向上を目的とした近似 LRU 実装です。 Redis には maxmemory-samples という構成があります。Redis の LRU は、構成された数のキーを取り出し、その中で最も使用頻度の低いキーを選択して置き換えます。デフォルトは次のように 5 です:maxmemory-samples 5
测试开始首先在Redis中导入一定数目的key,然后从第一个key依次访问到最后一个key,因此根据LRU算法第一个被访问的key应该最新被置换,之后再增加50%数目的key,导致50%的老的key被替换出去。
在上图中你可以看到三种类型的点,组成三种不同的区域:
淡灰色的是被置换出去的key
灰色的是没有被置换出去的key
绿色的是新增加的key
理论LRU实现就像我们期待的那样,最旧的50%数目的key被置换出去,Redis的LRU将一定比例的旧key置换出去。
可以看到在样本数为5的情况下,Redis3.0要比Redis2.8做的好很多,Redis2.8中有很多应该被置换出去的数据没有置换出去。在样本数为10的情况下,Redis3.0很接近真正的LRU实现。
LRU是一个预测未来我们会访问哪些数据的模型,如果我们访问数据的形式接近我们预想——幂律,那么近似LRU算法实现将能处理的很好。
在模拟测试中我们可以发现,在幂律访问模式下,理论LRU和Redis近似LRU的差距很小或者就不存在差距。
如果你将maxmemory-samples设置为10,那么Redis将会增加额外的CPU开销以保证接近真正的LRU性能,可以通过检查命中率来查看有什么不同。
通过CONFIG SET maxmemory-samples
2、LRU的实现
<?php /** * LRU是最近最少使用页面置换算法(Least Recently Used),也就是首先淘汰最长时间未被使用的页面 */ class LRU_Cache { private $array_lru = array(); private $max_size = 0; function __construct($size) { // 缓存最大存储 $this->max_size = $size; } public function set_value($key, $value) { // 如果存在,则向队尾移动,先删除,后追加 // array_key_exists() 函数检查某个数组中是否存在指定的键名,如果键名存在则返回true,如果键名不存在则返回false。 if (array_key_exists($key, $this->array_lru)) { // unset() 销毁指定的变量。 unset($this->array_lru[$key]); } // 长度检查,超长则删除首元素 if (count($this->array_lru) > $this->max_size) { // array_shift() 函数删除数组中第一个元素,并返回被删除元素的值。 array_shift($this->array_lru); } // 队尾追加元素 $this->array_lru[$key] = $value; } public function get_value($key) { $ret_value = false; if (array_key_exists($key, $this->array_lru)) { $ret_value = $this->array_lru[$key]; // 移动到队尾 unset($this->array_lru[$key]); $this->array_lru[$key] = $ret_value; } return $ret_value; } public function vardump_cache() { echo "<br>"; var_dump($this->array_lru); } } $cache = new LRU_Cache(5); // 指定了最大空间 6 $cache->set_value("01", "01"); $cache->set_value("02", "02"); $cache->set_value("03", "03"); $cache->set_value("04", "04"); $cache->set_value("05", "05"); $cache->vardump_cache(); echo "<br>"; $cache->set_value("06", "06"); $cache->vardump_cache(); echo "<br>"; $cache->set_value("03", "03"); $cache->vardump_cache(); echo "<br>"; $cache->set_value("07", "07"); $cache->vardump_cache(); echo "<br>"; $cache->set_value("01", "01"); $cache->vardump_cache(); echo "<br>"; $cache->get_value("04"); $cache->vardump_cache(); echo "<br>"; $cache->get_value("05"); $cache->vardump_cache(); echo "<br>"; $cache->get_value("10"); $cache->vardump_cache(); echo "<br>";
更多redis知识请关注redis入门教程栏目。
以上がRedis で LRU アルゴリズムを設定する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。