lru.go 5.7 KB

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