欧美阿v视频在线大全_亚洲欧美中文日韩V在线观看_www性欧美日韩欧美91_亚洲欧美日韩久久精品

主頁 > 知識庫 > Redis中LFU算法的深入分析

Redis中LFU算法的深入分析

熱門標簽:400電話辦理的口碑 臺灣電銷 b2b外呼系統 一個地圖標注多少錢 廊坊外呼系統在哪買 南京手機外呼系統廠家 高碑店市地圖標注app 四川穩定外呼系統軟件 地圖標注工廠入駐

前言

在Redis中的LRU算法文中說到,LRU有一個缺陷,在如下情況下:

~~~~~A~~~~~A~~~~~A~~~~A~~~~~A~~~~~A~~|
~~B~~B~~B~~B~~B~~B~~B~~B~~B~~B~~B~~B~|
~~~~~~~~~~C~~~~~~~~~C~~~~~~~~~C~~~~~~|
~~~~~D~~~~~~~~~~D~~~~~~~~~D~~~~~~~~~D|

會將數據D誤認為將來最有可能被訪問到的數據。

Redis作者曾想改進LRU算法,但發現Redis的LRU算法受制于隨機采樣數maxmemory_samples,在maxmemory_samples等于10的情況下已經很接近于理想的LRU算法性能,也就是說,LRU算法本身已經很難再進一步了。

于是,將思路回到原點,淘汰算法的本意是保留那些將來最有可能被再次訪問的數據,而LRU算法只是預測最近被訪問的數據將來最有可能被訪問到。我們可以轉變思路,采用一種LFU(Least Frequently Used)算法,也就是最頻繁被訪問的數據將來最有可能被訪問到。在上面的情況中,根據訪問頻繁情況,可以確定保留優先級:B>A>C=D。

Redis中的LFU思路

在LFU算法中,可以為每個key維護一個計數器。每次key被訪問的時候,計數器增大。計數器越大,可以約等于訪問越頻繁。

上述簡單算法存在兩個問題:

  • 在LRU算法中可以維護一個雙向鏈表,然后簡單的把被訪問的節點移至鏈表開頭,但在LFU中是不可行的,節點要嚴格按照計數器進行排序,新增節點或者更新節點位置時,時間復雜度可能達到O(N)。
  • 只是簡單的增加計數器的方法并不完美。訪問模式是會頻繁變化的,一段時間內頻繁訪問的key一段時間之后可能會很少被訪問到,只增加計數器并不能體現這種趨勢。

第一個問題很好解決,可以借鑒LRU實現的經驗,維護一個待淘汰key的pool。第二個問題的解決辦法是,記錄key最后一個被訪問的時間,然后隨著時間推移,降低計數器。

Redis對象的結構如下:

typedef struct redisObject {
  unsigned type:4;
  unsigned encoding:4;
  unsigned lru:LRU_BITS; /* LRU time (relative to global lru_clock) or
              * LFU data (least significant 8 bits frequency
              * and most significant 16 bits access time). */
  int refcount;
  void *ptr;
} robj;

在LRU算法中,24 bits的lru是用來記錄LRU time的,在LFU中也可以使用這個字段,不過是分成16 bits與8 bits使用:

      16 bits   8 bits
   +----------------+--------+
   + Last decr time | LOG_C |
   +----------------+--------+

高16 bits用來記錄最近一次計數器降低的時間ldt,單位是分鐘,低8 bits記錄計數器數值counter。

LFU配置

Redis4.0之后為maxmemory_policy淘汰策略添加了兩個LFU模式:

  • volatile-lfu:對有過期時間的key采用LFU淘汰算法
  • allkeys-lfu:對全部key采用LFU淘汰算法

還有2個配置可以調整LFU算法:

lfu-log-factor 10
lfu-decay-time 1

lfu-log-factor可以調整計數器counter的增長速度,lfu-log-factor越大,counter增長的越慢。

lfu-decay-time是一個以分鐘為單位的數值,可以調整counter的減少速度

源碼實現

在lookupKey中:

robj *lookupKey(redisDb *db, robj *key, int flags) {
  dictEntry *de = dictFind(db->dict,key->ptr);
  if (de) {
    robj *val = dictGetVal(de);

    /* Update the access time for the ageing algorithm.
     * Don't do it if we have a saving child, as this will trigger
     * a copy on write madness. */
    if (server.rdb_child_pid == -1 
      server.aof_child_pid == -1 
      !(flags  LOOKUP_NOTOUCH))
    {
      if (server.maxmemory_policy  MAXMEMORY_FLAG_LFU) {
        updateLFU(val);
      } else {
        val->lru = LRU_CLOCK();
      }
    }
    return val;
  } else {
    return NULL;
  }
}

當采用LFU策略時,updateLFU更新lru:

/* Update LFU when an object is accessed.
 * Firstly, decrement the counter if the decrement time is reached.
 * Then logarithmically increment the counter, and update the access time. */
void updateLFU(robj *val) {
  unsigned long counter = LFUDecrAndReturn(val);
  counter = LFULogIncr(counter);
  val->lru = (LFUGetTimeInMinutes()8) | counter;
}

降低LFUDecrAndReturn

首先,LFUDecrAndReturn對counter進行減少操作:

/* If the object decrement time is reached decrement the LFU counter but
 * do not update LFU fields of the object, we update the access time
 * and counter in an explicit way when the object is really accessed.
 * And we will times halve the counter according to the times of
 * elapsed time than server.lfu_decay_time.
 * Return the object frequency counter.
 *
 * This function is used in order to scan the dataset for the best object
 * to fit: as we check for the candidate, we incrementally decrement the
 * counter of the scanned objects if needed. */
unsigned long LFUDecrAndReturn(robj *o) {
  unsigned long ldt = o->lru >> 8;
  unsigned long counter = o->lru  255;
  unsigned long num_periods = server.lfu_decay_time ? LFUTimeElapsed(ldt) / server.lfu_decay_time : 0;
  if (num_periods)
    counter = (num_periods > counter) ? 0 : counter - num_periods;
  return counter;
}

函數首先取得高16 bits的最近降低時間ldt與低8 bits的計數器counter,然后根據配置的lfu_decay_time計算應該降低多少。

LFUTimeElapsed用來計算當前時間與ldt的差值:

/* Return the current time in minutes, just taking the least significant
 * 16 bits. The returned time is suitable to be stored as LDT (last decrement
 * time) for the LFU implementation. */
unsigned long LFUGetTimeInMinutes(void) {
  return (server.unixtime/60)  65535;
}

/* Given an object last access time, compute the minimum number of minutes
 * that elapsed since the last access. Handle overflow (ldt greater than
 * the current 16 bits minutes time) considering the time as wrapping
 * exactly once. */
unsigned long LFUTimeElapsed(unsigned long ldt) {
  unsigned long now = LFUGetTimeInMinutes();
  if (now >= ldt) return now-ldt;
  return 65535-ldt+now;
}

具體是當前時間轉化成分鐘數后取低16 bits,然后計算與ldt的差值now-ldt。當ldt > now時,默認為過了一個周期(16 bits,最大65535),取值65535-ldt+now。

然后用差值與配置lfu_decay_time相除,LFUTimeElapsed(ldt) / server.lfu_decay_time,已過去n個lfu_decay_time,則將counter減少n,counter - num_periods。

增長LFULogIncr

增長函數LFULogIncr如下:

/* Logarithmically increment a counter. The greater is the current counter value
 * the less likely is that it gets really implemented. Saturate it at 255. */
uint8_t LFULogIncr(uint8_t counter) {
  if (counter == 255) return 255;
  double r = (double)rand()/RAND_MAX;
  double baseval = counter - LFU_INIT_VAL;
  if (baseval  0) baseval = 0;
  double p = 1.0/(baseval*server.lfu_log_factor+1);
  if (r  p) counter++;
  return counter;
}

counter并不是簡單的訪問一次就+1,而是采用了一個0-1之間的p因子控制增長。counter最大值為255。取一個0-1之間的隨機數r與p比較,當rp時,才增加counter,這和比特幣中控制產出的策略類似。p取決于當前counter值與lfu_log_factor因子,counter值與lfu_log_factor因子越大,p越小,rp的概率也越小,counter增長的概率也就越小。增長情況如下:

+--------+------------+------------+------------+------------+------------+
| factor | 100 hits   | 1000 hits  | 100K hits  | 1M hits    | 10M hits   |
+--------+------------+------------+------------+------------+------------+
| 0      | 104        | 255        | 255        | 255        | 255        |
+--------+------------+------------+------------+------------+------------+
| 1      | 18         | 49         | 255        | 255        | 255        |
+--------+------------+------------+------------+------------+------------+
| 10     | 10         | 18         | 142        | 255        | 255        |
+--------+------------+------------+------------+------------+------------+
| 100    | 8          | 11         | 49         | 143        | 255        |
+--------+------------+------------+------------+------------+------------+

可見counter增長與訪問次數呈現對數增長的趨勢,隨著訪問次數越來越大,counter增長的越來越慢。

新生key策略

另外一個問題是,當創建新對象的時候,對象的counter如果為0,很容易就會被淘汰掉,還需要為新生key設置一個初始counter,createObject:

robj *createObject(int type, void *ptr) {
  robj *o = zmalloc(sizeof(*o));
  o->type = type;
  o->encoding = OBJ_ENCODING_RAW;
  o->ptr = ptr;
  o->refcount = 1;

  /* Set the LRU to the current lruclock (minutes resolution), or
   * alternatively the LFU counter. */
  if (server.maxmemory_policy  MAXMEMORY_FLAG_LFU) {
    o->lru = (LFUGetTimeInMinutes()8) | LFU_INIT_VAL;
  } else {
    o->lru = LRU_CLOCK();
  }
  return o;
}

counter會被初始化為LFU_INIT_VAL,默認5。

pool

pool算法就與LRU算法一致了:

    if (server.maxmemory_policy  (MAXMEMORY_FLAG_LRU|MAXMEMORY_FLAG_LFU) ||
      server.maxmemory_policy == MAXMEMORY_VOLATILE_TTL)

計算idle時有所不同:

    } else if (server.maxmemory_policy  MAXMEMORY_FLAG_LFU) {
      /* When we use an LRU policy, we sort the keys by idle time
       * so that we expire keys starting from greater idle time.
       * However when the policy is an LFU one, we have a frequency
       * estimation, and we want to evict keys with lower frequency
       * first. So inside the pool we put objects using the inverted
       * frequency subtracting the actual frequency to the maximum
       * frequency of 255. */
      idle = 255-LFUDecrAndReturn(o);

使用了255-LFUDecrAndReturn(o)當做排序的依據。

參考鏈接

  • Random notes on improving the Redis LRU algorithm
  • Using Redis as an LRU cache

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • C++ 實現LRU 與 LFU 的緩存算法
  • C++泛型編程基本概念詳解
  • C++算法與泛型算法(algorithm、numeric)
  • C++ 泛型編程詳解
  • C++實現支持泛型的LFU詳解

標簽:伊春 南寧 定州 河源 泰州 拉薩 甘南 畢節

巨人網絡通訊聲明:本文標題《Redis中LFU算法的深入分析》,本文關鍵詞  Redis,中,LFU,算法,的,深入分析,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Redis中LFU算法的深入分析》相關的同類信息!
  • 本頁收集關于Redis中LFU算法的深入分析的相關信息資訊供網民參考!
  • 推薦文章
    欧美阿v视频在线大全_亚洲欧美中文日韩V在线观看_www性欧美日韩欧美91_亚洲欧美日韩久久精品
  • <rt id="w000q"><acronym id="w000q"></acronym></rt>
  • <abbr id="w000q"></abbr>
    <rt id="w000q"></rt>
    日本少妇激三级做爰在线| 日本一区二区不卡视频| 欧美日韩一区二区在线观看视频| 在线视频你懂得一区二区三区| 欧美在线一二三四区| 538在线一区二区精品国产| 欧美mv日韩mv| 国产精品伦理一区二区| 亚洲精品国产精华液| 婷婷开心激情综合| 国产在线不卡视频| 91一区二区三区在线播放| 中文字幕免费在线播放| 国产破处视频在线观看| 欧美优质美女网站| www久久精品| 亚洲免费看黄网站| 久久99精品久久只有精品| 不卡的看片网站| 色噜噜在线观看| 久久国产精品国语对白| 91精品国产色综合久久不卡蜜臀| 久久久久国产精品人| 一区二区三区**美女毛片| 九色porny丨国产精品| 久久久久久国产精品日本| www.黄色在线| 欧美日韩极品在线观看一区| 国产香蕉久久精品综合网| 亚洲主播在线观看| 国产一区二区三区四| 国产伦理在线观看| 激情无码人妻又粗又大| 欧美一级二级三级蜜桃| 日韩美女视频一区| 美女诱惑一区二区| 美女流白浆视频| 国产免费久久久久| 欧美成人高清电影在线| 亚洲精品久久嫩草网站秘色| 狠狠色狠狠色综合系列| 精品国产乱码久久久久夜深人妻| 亚洲综合久久av一区二区三区| 91精品国产综合久久精品app| 亚洲人妖av一区二区| 国产精品一级二级三级| 不卡一区二区在线观看| 国产传媒欧美日韩成人| 亚洲av成人精品一区二区三区| 一级片一级片一级片| 精品久久免费看| 图片区小说区国产精品视频| 不卡av电影在线播放| 欧美激情 一区| 精品日韩一区二区三区免费视频| 一区二区在线观看免费| 岛国精品一区二区| www亚洲色图| 欧美成人欧美edvon| 亚洲不卡在线观看| 日韩精品xxx| 色香色香欲天天天影视综合网 | 日韩欧美国产一区在线观看| 一区二区三区影院| thepron国产精品| 男人晚上看的视频| 久久久99精品免费观看| 毛片一区二区三区| 少妇饥渴放荡91麻豆| 欧美男女性生活在线直播观看 | 久久99精品一区二区三区三区| 中文字幕在线播放视频| 欧美日韩你懂得| 亚洲一区二区三区小说| 性高潮久久久久久| 欧美探花视频资源| 亚洲永久免费视频| av电影中文字幕| 欧美日韩日日摸| 亚洲bt欧美bt精品| 精品人妻一区二区三区日产| 欧美人动与zoxxxx乱| 亚洲国产cao| 国产午夜在线一区二区三区| 欧美日本国产视频| 五月婷婷久久丁香| 狠狠人妻久久久久久综合蜜桃| 日韩一区二区三区四区| 麻豆精品在线看| 欧美性受xxxx黑人| 国产精品水嫩水嫩| www.亚洲人| 欧美日韩高清不卡| 日韩电影免费在线| 免费观看a级片| 国产亲近乱来精品视频| 成人黄色av网站在线| 色婷婷国产精品| 亚洲香肠在线观看| 免费在线观看成年人视频| 久久综合色婷婷| 国产suv一区二区三区88区| 亚洲欧美一区二区三区四区五区| 亚洲精品免费看| 欧美xxxxx精品| 2022国产精品视频| 国产传媒欧美日韩成人| 色噜噜狠狠成人网p站| 亚洲国产精品一区二区久久| 蜜桃传媒一区二区亚洲av| 国产午夜三级一区二区三| 成人高清伦理免费影院在线观看| 欧美性xxxxxxxx| 青青草国产成人99久久| 青青草华人在线视频| 综合在线观看色| 2一3sex性hd| 久久精品人人做| 99精品在线观看视频| 91精品国产一区二区| 国产在线精品一区二区| 色综合天天狠狠| 日韩精品久久久久久| 人与动物性xxxx| 一区二区在线观看不卡| 伊人网在线视频观看| 中文字幕一区二区三区蜜月 | 91嫩草|国产丨精品入口| 亚洲欧美激情在线| 五月开心播播网| 国产精品亲子伦对白| 精品伦一区二区三区| 久久人人97超碰com| 99国产精品久久久久久久久久| 91麻豆精品国产91久久久使用方法 | 亚洲大片一区二区三区| xxxx日本黄色| 一区二区三区日本| 级毛片内射视频| 一区二区三区免费观看| 非洲一级黄色片| 亚洲最新在线观看| 黄色裸体一级片| 图片区小说区区亚洲影院| 丝袜美腿小色网| 免费在线观看不卡| 91福利在线播放| 国产酒店精品激情| 欧美一级日韩一级| 91丨porny丨国产入口| 久久久久久免费网| 性久久久久久久久久久| 亚洲人成在线播放网站岛国| 老熟妇一区二区| 亚洲成人精品一区二区| 亚洲综合网在线| 久久国内精品自在自线400部| 在线一区二区三区做爰视频网站| 国内精品嫩模私拍在线| 7777精品伊人久久久大香线蕉完整版 | 亚洲图片另类小说| 女女互磨互喷水高潮les呻吟| 亚洲国产裸拍裸体视频在线观看乱了 | 亚洲精品在线电影| 亚洲女则毛耸耸bbw| 中文字幕亚洲欧美在线不卡| 摸摸摸bbb毛毛毛片| 视频在线观看一区二区三区| 色老汉一区二区三区| 国产不卡免费视频| 久久综合久久鬼色中文字| 荫蒂被男人添免费视频| 一区二区三区在线高清| 日本中文字幕免费在线观看| 国产麻豆视频精品| 精品av久久707| 日本黄色动态图| 亚洲成av人片观看| 欧美无乱码久久久免费午夜一区| 粉嫩一区二区三区性色av| 久久久久综合网| 国产精品密蕾丝袜| 青娱乐精品视频在线| 欧美一区国产二区| 大尺度做爰床戏呻吟舒畅| 亚洲国产日韩精品| 欧美喷水一区二区| 年下总裁被打光屁股sp| 一区二区三区四区激情| 色婷婷久久久久swag精品| 成人av电影免费在线播放| 国产精品久久久久精k8| 中文字幕第69页| 国产精品综合在线视频| 久久久久久黄色| 超碰人人干人人| 国产制服丝袜一区| 国产亚洲制服色| 午夜国产福利视频| 国产成人av自拍|