cachemap.go 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. package cache
  2. import (
  3. "runtime"
  4. "sync"
  5. "time"
  6. )
  7. type CallCreate func(key interface{}) (value interface{}, created bool)
  8. type CallCreateNew func(key interface{}) (rKey, value interface{}, created bool)
  9. type CallCheck func(key, value interface{}, exists bool) (rKey, rValue interface{}, created bool)
  10. func NewMap(liveDuration time.Duration, maxSize int) *CacheMap {
  11. res := &CacheMap{&cacheMap{
  12. locker: new(sync.RWMutex),
  13. items: make(map[interface{}]*cacheMapItem),
  14. liveDuration: liveDuration,
  15. maxSize: maxSize,
  16. stopCleanerChan: make(chan byte),
  17. }}
  18. runtime.SetFinalizer(res, destroyCacheMap)
  19. return res
  20. }
  21. type CacheMap struct {
  22. *cacheMap
  23. }
  24. type cacheMapItem struct {
  25. value interface{}
  26. expire int64
  27. }
  28. type cacheMap struct {
  29. locker *sync.RWMutex
  30. items map[interface{}]*cacheMapItem
  31. liveDuration time.Duration
  32. maxSize int
  33. cleanerWork bool
  34. stopCleanerChan chan byte
  35. cleanedEventChan chan map[interface{}]interface{}
  36. }
  37. func (s *cacheMap) CleanEvent() (eventChan <-chan map[interface{}]interface{}) {
  38. s.locker.Lock()
  39. if s.cleanedEventChan == nil {
  40. s.cleanedEventChan = make(chan map[interface{}]interface{})
  41. }
  42. eventChan = s.cleanedEventChan
  43. s.locker.Unlock()
  44. return
  45. }
  46. func (s *cacheMap) runCleaner() {
  47. ticker := time.NewTicker(s.liveDuration / 2)
  48. for {
  49. select {
  50. case <-ticker.C:
  51. {
  52. now := time.Now().UnixNano()
  53. s.locker.Lock()
  54. cleaned := make(map[interface{}]interface{})
  55. for key, val := range s.items {
  56. if now > val.expire {
  57. cleaned[key] = val.value
  58. delete(s.items, key)
  59. }
  60. }
  61. if len(cleaned) > 0 {
  62. select {
  63. case s.cleanedEventChan <- cleaned:
  64. default:
  65. }
  66. }
  67. if len(s.items) == 0 {
  68. s.cleanerWork = false
  69. ticker.Stop()
  70. s.locker.Unlock()
  71. return
  72. }
  73. s.locker.Unlock()
  74. }
  75. case <-s.stopCleanerChan:
  76. {
  77. s.cleanerWork = false
  78. ticker.Stop()
  79. return
  80. }
  81. }
  82. }
  83. }
  84. func (s *cacheMap) delete(key interface{}) {
  85. delete(s.items, key)
  86. if len(s.items) == 0 && s.cleanerWork {
  87. s.stopCleanerChan <- 0
  88. }
  89. }
  90. // Delete removes cached object
  91. func (s *cacheMap) Delete(key interface{}) {
  92. s.locker.Lock()
  93. s.delete(key)
  94. s.locker.Unlock()
  95. }
  96. func (s *cacheMap) set(key, value interface{}) {
  97. s.items[key] = &cacheMapItem{
  98. value: value,
  99. expire: time.Now().Add(s.liveDuration).UnixNano(),
  100. }
  101. if !s.cleanerWork {
  102. s.cleanerWork = true
  103. go s.runCleaner()
  104. }
  105. }
  106. func (s *cacheMap) Set(key, value interface{}) {
  107. s.locker.Lock()
  108. s.set(key, value)
  109. s.locker.Unlock()
  110. }
  111. func (s *cacheMap) SetMulti(m map[interface{}]interface{}) {
  112. s.locker.Lock()
  113. for key, val := range m {
  114. s.set(key, val)
  115. }
  116. s.locker.Unlock()
  117. }
  118. func (s *CacheMap) DeleteMulti(keys []interface{}) {
  119. s.locker.Lock()
  120. for _, key := range keys {
  121. delete(s.items, key)
  122. }
  123. s.locker.Unlock()
  124. }
  125. func (s *cacheMap) get(key interface{}) (res interface{}, check bool) {
  126. var item *cacheMapItem
  127. if item, check = s.items[key]; check {
  128. res = item.value
  129. }
  130. return
  131. }
  132. func (s *cacheMap) Get(key interface{}) (res interface{}, check bool) {
  133. s.locker.RLock()
  134. res, check = s.get(key)
  135. s.locker.RUnlock()
  136. return
  137. }
  138. func (s *cacheMap) GetCreate(key interface{}, mCreate CallCreate) (val interface{}, check bool) {
  139. if val, check = s.Get(key); !check {
  140. s.locker.Lock()
  141. if val, check = s.get(key); check {
  142. s.locker.Unlock()
  143. return
  144. }
  145. if val, check = mCreate(key); check {
  146. s.set(key, val)
  147. }
  148. s.locker.Unlock()
  149. }
  150. return
  151. }
  152. func (s *cacheMap) GetCreateNew(key interface{}, mCreateNew CallCreateNew) (rKey, val interface{}, check bool) {
  153. rKey = key
  154. if val, check = s.Get(key); !check {
  155. s.locker.Lock()
  156. if val, check = s.get(key); check {
  157. s.locker.Unlock()
  158. return
  159. }
  160. if rKey, val, check = mCreateNew(key); check {
  161. s.set(rKey, val)
  162. }
  163. s.locker.Unlock()
  164. }
  165. return
  166. }
  167. func (s *cacheMap) GetCheck(key interface{}, mCheck CallCheck) (res interface{}, check bool) {
  168. s.locker.Lock()
  169. res, check = s.get(key)
  170. if rKey, rVal, rCheck := mCheck(key, res, check); rCheck {
  171. s.set(rKey, rVal)
  172. key, res, check = rKey, rVal, true
  173. }
  174. s.locker.Unlock()
  175. return
  176. }
  177. // Each implements a map bypass for each key using the callback function. If the callback function returns false, then the cycle stops
  178. func (s *cacheMap) Each(callback func(interface{}, interface{}) bool) {
  179. s.locker.RLock()
  180. for key, val := range s.items {
  181. if !callback(key, val.value) {
  182. s.locker.RUnlock()
  183. return
  184. }
  185. }
  186. s.locker.RUnlock()
  187. }
  188. func (s *cacheMap) Len() (res int) {
  189. s.locker.RLock()
  190. res = len(s.items)
  191. s.locker.RUnlock()
  192. return
  193. }
  194. func (s *cacheMap) Keys() (res []interface{}) {
  195. s.locker.RLock()
  196. res = make([]interface{}, 0, len(s.items))
  197. for key := range s.items {
  198. res = append(res, key)
  199. }
  200. s.locker.RUnlock()
  201. return
  202. }
  203. func (s *cacheMap) Clear() {
  204. s.locker.Lock()
  205. s.items = make(map[interface{}]*cacheMapItem)
  206. s.locker.Unlock()
  207. }
  208. // for garbage collector
  209. func destroyCacheMap(m *CacheMap) {
  210. close(m.stopCleanerChan)
  211. if m.cleanedEventChan != nil {
  212. close(m.cleanedEventChan)
  213. }
  214. }