lfu.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. package gcache
  2. import (
  3. "container/list"
  4. "time"
  5. )
  6. // Discards the least frequently used items first.
  7. type LFUCache struct {
  8. baseCache
  9. items map[interface{}]*lfuItem
  10. freqList *list.List // list for freqEntry
  11. }
  12. func newLFUCache(cb *CacheBuilder) *LFUCache {
  13. c := &LFUCache{}
  14. buildCache(&c.baseCache, cb)
  15. c.init()
  16. c.loadGroup.cache = c
  17. return c
  18. }
  19. func (c *LFUCache) init() {
  20. c.freqList = list.New()
  21. c.items = make(map[interface{}]*lfuItem, c.size+1)
  22. c.freqList.PushFront(&freqEntry{
  23. freq: 0,
  24. items: make(map[*lfuItem]struct{}),
  25. })
  26. }
  27. // Set a new key-value pair
  28. func (c *LFUCache) Set(key, value interface{}) error {
  29. c.mu.Lock()
  30. defer c.mu.Unlock()
  31. _, err := c.set(key, value)
  32. return err
  33. }
  34. // Set a new key-value pair with an expiration time
  35. func (c *LFUCache) SetWithExpire(key, value interface{}, expiration time.Duration) error {
  36. c.mu.Lock()
  37. defer c.mu.Unlock()
  38. item, err := c.set(key, value)
  39. if err != nil {
  40. return err
  41. }
  42. t := c.clock.Now().Add(expiration)
  43. item.(*lfuItem).expiration = &t
  44. return nil
  45. }
  46. func (c *LFUCache) set(key, value interface{}) (interface{}, error) {
  47. var err error
  48. if c.serializeFunc != nil {
  49. value, err = c.serializeFunc(key, value)
  50. if err != nil {
  51. return nil, err
  52. }
  53. }
  54. // Check for existing item
  55. item, ok := c.items[key]
  56. if ok {
  57. item.value = value
  58. } else {
  59. // Verify size not exceeded
  60. if len(c.items) >= c.size {
  61. c.evict(1)
  62. }
  63. item = &lfuItem{
  64. clock: c.clock,
  65. key: key,
  66. value: value,
  67. freqElement: nil,
  68. }
  69. el := c.freqList.Front()
  70. fe := el.Value.(*freqEntry)
  71. fe.items[item] = struct{}{}
  72. item.freqElement = el
  73. c.items[key] = item
  74. }
  75. if c.expiration != nil {
  76. t := c.clock.Now().Add(*c.expiration)
  77. item.expiration = &t
  78. }
  79. if c.addedFunc != nil {
  80. c.addedFunc(key, value)
  81. }
  82. return item, nil
  83. }
  84. // Get a value from cache pool using key if it exists.
  85. // If it dose not exists key and has LoaderFunc,
  86. // generate a value using `LoaderFunc` method returns value.
  87. func (c *LFUCache) Get(key interface{}) (interface{}, error) {
  88. v, err := c.get(key, false)
  89. if err == KeyNotFoundError {
  90. return c.getWithLoader(key, true)
  91. }
  92. return v, err
  93. }
  94. // Get a value from cache pool using key if it exists.
  95. // If it dose not exists key, returns KeyNotFoundError.
  96. // And send a request which refresh value for specified key if cache object has LoaderFunc.
  97. func (c *LFUCache) GetIFPresent(key interface{}) (interface{}, error) {
  98. v, err := c.get(key, false)
  99. if err == KeyNotFoundError {
  100. return c.getWithLoader(key, false)
  101. }
  102. return v, err
  103. }
  104. func (c *LFUCache) get(key interface{}, onLoad bool) (interface{}, error) {
  105. v, err := c.getValue(key, onLoad)
  106. if err != nil {
  107. return nil, err
  108. }
  109. if c.deserializeFunc != nil {
  110. return c.deserializeFunc(key, v)
  111. }
  112. return v, nil
  113. }
  114. func (c *LFUCache) getValue(key interface{}, onLoad bool) (interface{}, error) {
  115. c.mu.Lock()
  116. item, ok := c.items[key]
  117. if ok {
  118. if !item.IsExpired(nil) {
  119. c.increment(item)
  120. v := item.value
  121. c.mu.Unlock()
  122. if !onLoad {
  123. c.stats.IncrHitCount()
  124. }
  125. return v, nil
  126. }
  127. c.removeItem(item)
  128. }
  129. c.mu.Unlock()
  130. if !onLoad {
  131. c.stats.IncrMissCount()
  132. }
  133. return nil, KeyNotFoundError
  134. }
  135. func (c *LFUCache) getWithLoader(key interface{}, isWait bool) (interface{}, error) {
  136. if c.loaderExpireFunc == nil {
  137. return nil, KeyNotFoundError
  138. }
  139. value, _, err := c.load(key, func(v interface{}, expiration *time.Duration, e error) (interface{}, error) {
  140. if e != nil {
  141. return nil, e
  142. }
  143. c.mu.Lock()
  144. defer c.mu.Unlock()
  145. item, err := c.set(key, v)
  146. if err != nil {
  147. return nil, err
  148. }
  149. if expiration != nil {
  150. t := c.clock.Now().Add(*expiration)
  151. item.(*lfuItem).expiration = &t
  152. }
  153. return v, nil
  154. }, isWait)
  155. if err != nil {
  156. return nil, err
  157. }
  158. return value, nil
  159. }
  160. func (c *LFUCache) increment(item *lfuItem) {
  161. currentFreqElement := item.freqElement
  162. currentFreqEntry := currentFreqElement.Value.(*freqEntry)
  163. nextFreq := currentFreqEntry.freq + 1
  164. delete(currentFreqEntry.items, item)
  165. nextFreqElement := currentFreqElement.Next()
  166. if nextFreqElement == nil {
  167. nextFreqElement = c.freqList.InsertAfter(&freqEntry{
  168. freq: nextFreq,
  169. items: make(map[*lfuItem]struct{}),
  170. }, currentFreqElement)
  171. }
  172. nextFreqElement.Value.(*freqEntry).items[item] = struct{}{}
  173. item.freqElement = nextFreqElement
  174. }
  175. // evict removes the least frequence item from the cache.
  176. func (c *LFUCache) evict(count int) {
  177. entry := c.freqList.Front()
  178. for i := 0; i < count; {
  179. if entry == nil {
  180. return
  181. } else {
  182. for item, _ := range entry.Value.(*freqEntry).items {
  183. if i >= count {
  184. return
  185. }
  186. c.removeItem(item)
  187. i++
  188. }
  189. entry = entry.Next()
  190. }
  191. }
  192. }
  193. // Removes the provided key from the cache.
  194. func (c *LFUCache) Remove(key interface{}) bool {
  195. c.mu.Lock()
  196. defer c.mu.Unlock()
  197. return c.remove(key)
  198. }
  199. func (c *LFUCache) remove(key interface{}) bool {
  200. if item, ok := c.items[key]; ok {
  201. c.removeItem(item)
  202. return true
  203. }
  204. return false
  205. }
  206. // removeElement is used to remove a given list element from the cache
  207. func (c *LFUCache) removeItem(item *lfuItem) {
  208. delete(c.items, item.key)
  209. delete(item.freqElement.Value.(*freqEntry).items, item)
  210. if c.evictedFunc != nil {
  211. c.evictedFunc(item.key, item.value)
  212. }
  213. }
  214. func (c *LFUCache) keys() []interface{} {
  215. c.mu.RLock()
  216. defer c.mu.RUnlock()
  217. keys := make([]interface{}, len(c.items))
  218. var i = 0
  219. for k := range c.items {
  220. keys[i] = k
  221. i++
  222. }
  223. return keys
  224. }
  225. // Returns a slice of the keys in the cache.
  226. func (c *LFUCache) Keys() []interface{} {
  227. keys := []interface{}{}
  228. for _, k := range c.keys() {
  229. _, err := c.GetIFPresent(k)
  230. if err == nil {
  231. keys = append(keys, k)
  232. }
  233. }
  234. return keys
  235. }
  236. // Returns all key-value pairs in the cache.
  237. func (c *LFUCache) GetALL() map[interface{}]interface{} {
  238. m := make(map[interface{}]interface{})
  239. for _, k := range c.keys() {
  240. v, err := c.GetIFPresent(k)
  241. if err == nil {
  242. m[k] = v
  243. }
  244. }
  245. return m
  246. }
  247. // Returns the number of items in the cache.
  248. func (c *LFUCache) Len() int {
  249. return len(c.GetALL())
  250. }
  251. // Completely clear the cache
  252. func (c *LFUCache) Purge() {
  253. c.mu.Lock()
  254. defer c.mu.Unlock()
  255. if c.purgeVisitorFunc != nil {
  256. for key, item := range c.items {
  257. c.purgeVisitorFunc(key, item.value)
  258. }
  259. }
  260. c.init()
  261. }
  262. type freqEntry struct {
  263. freq uint
  264. items map[*lfuItem]struct{}
  265. }
  266. type lfuItem struct {
  267. clock Clock
  268. key interface{}
  269. value interface{}
  270. freqElement *list.Element
  271. expiration *time.Time
  272. }
  273. // returns boolean value whether this item is expired or not.
  274. func (it *lfuItem) IsExpired(now *time.Time) bool {
  275. if it.expiration == nil {
  276. return false
  277. }
  278. if now == nil {
  279. t := it.clock.Now()
  280. now = &t
  281. }
  282. return it.expiration.Before(*now)
  283. }