arc.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. package gcache
  2. import (
  3. "container/list"
  4. "time"
  5. )
  6. // Constantly balances between LRU and LFU, to improve the combined result.
  7. type ARC struct {
  8. baseCache
  9. items map[interface{}]*arcItem
  10. part int
  11. t1 *arcList
  12. t2 *arcList
  13. b1 *arcList
  14. b2 *arcList
  15. }
  16. func newARC(cb *CacheBuilder) *ARC {
  17. c := &ARC{}
  18. buildCache(&c.baseCache, cb)
  19. c.init()
  20. c.loadGroup.cache = c
  21. return c
  22. }
  23. func (c *ARC) init() {
  24. c.items = make(map[interface{}]*arcItem)
  25. c.t1 = newARCList()
  26. c.t2 = newARCList()
  27. c.b1 = newARCList()
  28. c.b2 = newARCList()
  29. }
  30. func (c *ARC) replace(key interface{}) {
  31. var old interface{}
  32. if (c.t1.Len() > 0 && c.b2.Has(key) && c.t1.Len() == c.part) || (c.t1.Len() > c.part) {
  33. old = c.t1.RemoveTail()
  34. c.b1.PushFront(old)
  35. } else if c.t2.l.Len() > 0 {
  36. old = c.t2.RemoveTail()
  37. c.b2.PushFront(old)
  38. } else {
  39. return
  40. }
  41. item, ok := c.items[old]
  42. if ok {
  43. delete(c.items, old)
  44. if c.evictedFunc != nil {
  45. c.evictedFunc(item.key, item.value)
  46. }
  47. }
  48. }
  49. func (c *ARC) Set(key, value interface{}) error {
  50. c.mu.Lock()
  51. defer c.mu.Unlock()
  52. _, err := c.set(key, value)
  53. return err
  54. }
  55. // Set a new key-value pair with an expiration time
  56. func (c *ARC) SetWithExpire(key, value interface{}, expiration time.Duration) error {
  57. c.mu.Lock()
  58. defer c.mu.Unlock()
  59. item, err := c.set(key, value)
  60. if err != nil {
  61. return err
  62. }
  63. t := c.clock.Now().Add(expiration)
  64. item.(*arcItem).expiration = &t
  65. return nil
  66. }
  67. func (c *ARC) set(key, value interface{}) (interface{}, error) {
  68. var err error
  69. if c.serializeFunc != nil {
  70. value, err = c.serializeFunc(key, value)
  71. if err != nil {
  72. return nil, err
  73. }
  74. }
  75. item, ok := c.items[key]
  76. if ok {
  77. item.value = value
  78. } else {
  79. item = &arcItem{
  80. clock: c.clock,
  81. key: key,
  82. value: value,
  83. }
  84. c.items[key] = item
  85. }
  86. if c.expiration != nil {
  87. t := c.clock.Now().Add(*c.expiration)
  88. item.expiration = &t
  89. }
  90. defer func() {
  91. if c.addedFunc != nil {
  92. c.addedFunc(key, value)
  93. }
  94. }()
  95. if c.t1.Has(key) || c.t2.Has(key) {
  96. return item, nil
  97. }
  98. if elt := c.b1.Lookup(key); elt != nil {
  99. c.part = minInt(c.size, c.part+maxInt(c.b2.Len()/c.b1.Len(), 1))
  100. c.replace(key)
  101. c.b1.Remove(key, elt)
  102. c.t2.PushFront(key)
  103. return item, nil
  104. }
  105. if elt := c.b2.Lookup(key); elt != nil {
  106. c.part = maxInt(0, c.part-maxInt(c.b1.Len()/c.b2.Len(), 1))
  107. c.replace(key)
  108. c.b2.Remove(key, elt)
  109. c.t2.PushFront(key)
  110. return item, nil
  111. }
  112. if c.t1.Len()+c.b1.Len() == c.size {
  113. if c.t1.Len() < c.size {
  114. c.b1.RemoveTail()
  115. c.replace(key)
  116. } else {
  117. pop := c.t1.RemoveTail()
  118. item, ok := c.items[pop]
  119. if ok {
  120. delete(c.items, pop)
  121. if c.evictedFunc != nil {
  122. c.evictedFunc(item.key, item.value)
  123. }
  124. }
  125. }
  126. } else {
  127. total := c.t1.Len() + c.b1.Len() + c.t2.Len() + c.b2.Len()
  128. if total >= c.size {
  129. if total == (2 * c.size) {
  130. c.b2.RemoveTail()
  131. }
  132. c.replace(key)
  133. }
  134. }
  135. c.t1.PushFront(key)
  136. return item, nil
  137. }
  138. // Get a value from cache pool using key if it exists. If not exists and it has LoaderFunc, it will generate the value using you have specified LoaderFunc method returns value.
  139. func (c *ARC) Get(key interface{}) (interface{}, error) {
  140. v, err := c.get(key, false)
  141. if err == KeyNotFoundError {
  142. return c.getWithLoader(key, true)
  143. }
  144. return v, err
  145. }
  146. // Get a value from cache pool using key if it exists.
  147. // If it dose not exists key, returns KeyNotFoundError.
  148. // And send a request which refresh value for specified key if cache object has LoaderFunc.
  149. func (c *ARC) GetIFPresent(key interface{}) (interface{}, error) {
  150. v, err := c.get(key, false)
  151. if err == KeyNotFoundError {
  152. return c.getWithLoader(key, false)
  153. }
  154. return v, err
  155. }
  156. func (c *ARC) get(key interface{}, onLoad bool) (interface{}, error) {
  157. v, err := c.getValue(key, onLoad)
  158. if err != nil {
  159. return nil, err
  160. }
  161. if c.deserializeFunc != nil {
  162. return c.deserializeFunc(key, v)
  163. }
  164. return v, nil
  165. }
  166. func (c *ARC) getValue(key interface{}, onLoad bool) (interface{}, error) {
  167. c.mu.Lock()
  168. defer c.mu.Unlock()
  169. if elt := c.t1.Lookup(key); elt != nil {
  170. c.t1.Remove(key, elt)
  171. item := c.items[key]
  172. if !item.IsExpired(nil) {
  173. c.t2.PushFront(key)
  174. if !onLoad {
  175. c.stats.IncrHitCount()
  176. }
  177. return item.value, nil
  178. } else {
  179. delete(c.items, key)
  180. c.b1.PushFront(key)
  181. if c.evictedFunc != nil {
  182. c.evictedFunc(item.key, item.value)
  183. }
  184. }
  185. }
  186. if elt := c.t2.Lookup(key); elt != nil {
  187. item := c.items[key]
  188. if !item.IsExpired(nil) {
  189. c.t2.MoveToFront(elt)
  190. if !onLoad {
  191. c.stats.IncrHitCount()
  192. }
  193. return item.value, nil
  194. } else {
  195. delete(c.items, key)
  196. c.t2.Remove(key, elt)
  197. c.b2.PushFront(key)
  198. if c.evictedFunc != nil {
  199. c.evictedFunc(item.key, item.value)
  200. }
  201. }
  202. }
  203. if !onLoad {
  204. c.stats.IncrMissCount()
  205. }
  206. return nil, KeyNotFoundError
  207. }
  208. func (c *ARC) getWithLoader(key interface{}, isWait bool) (interface{}, error) {
  209. if c.loaderExpireFunc == nil {
  210. return nil, KeyNotFoundError
  211. }
  212. value, _, err := c.load(key, func(v interface{}, expiration *time.Duration, e error) (interface{}, error) {
  213. if e != nil {
  214. return nil, e
  215. }
  216. c.mu.Lock()
  217. defer c.mu.Unlock()
  218. item, err := c.set(key, v)
  219. if err != nil {
  220. return nil, err
  221. }
  222. if expiration != nil {
  223. t := c.clock.Now().Add(*expiration)
  224. item.(*arcItem).expiration = &t
  225. }
  226. return v, nil
  227. }, isWait)
  228. if err != nil {
  229. return nil, err
  230. }
  231. return value, nil
  232. }
  233. // Remove removes the provided key from the cache.
  234. func (c *ARC) Remove(key interface{}) bool {
  235. c.mu.Lock()
  236. defer c.mu.Unlock()
  237. return c.remove(key)
  238. }
  239. func (c *ARC) remove(key interface{}) bool {
  240. if elt := c.t1.Lookup(key); elt != nil {
  241. c.t1.Remove(key, elt)
  242. item := c.items[key]
  243. delete(c.items, key)
  244. c.b1.PushFront(key)
  245. if c.evictedFunc != nil {
  246. c.evictedFunc(key, item.value)
  247. }
  248. return true
  249. }
  250. if elt := c.t2.Lookup(key); elt != nil {
  251. c.t2.Remove(key, elt)
  252. item := c.items[key]
  253. delete(c.items, key)
  254. c.b2.PushFront(key)
  255. if c.evictedFunc != nil {
  256. c.evictedFunc(key, item.value)
  257. }
  258. return true
  259. }
  260. return false
  261. }
  262. func (c *ARC) keys() []interface{} {
  263. c.mu.RLock()
  264. defer c.mu.RUnlock()
  265. keys := make([]interface{}, len(c.items))
  266. var i = 0
  267. for k := range c.items {
  268. keys[i] = k
  269. i++
  270. }
  271. return keys
  272. }
  273. // Keys returns a slice of the keys in the cache.
  274. func (c *ARC) Keys() []interface{} {
  275. keys := []interface{}{}
  276. for _, k := range c.keys() {
  277. _, err := c.GetIFPresent(k)
  278. if err == nil {
  279. keys = append(keys, k)
  280. }
  281. }
  282. return keys
  283. }
  284. // Returns all key-value pairs in the cache.
  285. func (c *ARC) GetALL() map[interface{}]interface{} {
  286. m := make(map[interface{}]interface{})
  287. for _, k := range c.keys() {
  288. v, err := c.GetIFPresent(k)
  289. if err == nil {
  290. m[k] = v
  291. }
  292. }
  293. return m
  294. }
  295. // Len returns the number of items in the cache.
  296. func (c *ARC) Len() int {
  297. return len(c.GetALL())
  298. }
  299. // Purge is used to completely clear the cache
  300. func (c *ARC) Purge() {
  301. c.mu.Lock()
  302. defer c.mu.Unlock()
  303. if c.purgeVisitorFunc != nil {
  304. for _, item := range c.items {
  305. c.purgeVisitorFunc(item.key, item.value)
  306. }
  307. }
  308. c.init()
  309. }
  310. // returns boolean value whether this item is expired or not.
  311. func (it *arcItem) IsExpired(now *time.Time) bool {
  312. if it.expiration == nil {
  313. return false
  314. }
  315. if now == nil {
  316. t := it.clock.Now()
  317. now = &t
  318. }
  319. return it.expiration.Before(*now)
  320. }
  321. type arcList struct {
  322. l *list.List
  323. keys map[interface{}]*list.Element
  324. }
  325. type arcItem struct {
  326. clock Clock
  327. key interface{}
  328. value interface{}
  329. expiration *time.Time
  330. }
  331. func newARCList() *arcList {
  332. return &arcList{
  333. l: list.New(),
  334. keys: make(map[interface{}]*list.Element),
  335. }
  336. }
  337. func (al *arcList) Has(key interface{}) bool {
  338. _, ok := al.keys[key]
  339. return ok
  340. }
  341. func (al *arcList) Lookup(key interface{}) *list.Element {
  342. elt := al.keys[key]
  343. return elt
  344. }
  345. func (al *arcList) MoveToFront(elt *list.Element) {
  346. al.l.MoveToFront(elt)
  347. }
  348. func (al *arcList) PushFront(key interface{}) {
  349. if elt, ok := al.keys[key]; ok {
  350. al.l.MoveToFront(elt)
  351. return
  352. }
  353. elt := al.l.PushFront(key)
  354. al.keys[key] = elt
  355. }
  356. func (al *arcList) Remove(key interface{}, elt *list.Element) {
  357. delete(al.keys, key)
  358. al.l.Remove(elt)
  359. }
  360. func (al *arcList) RemoveTail() interface{} {
  361. elt := al.l.Back()
  362. al.l.Remove(elt)
  363. key := elt.Value
  364. delete(al.keys, key)
  365. return key
  366. }
  367. func (al *arcList) Len() int {
  368. return al.l.Len()
  369. }