LRUCache.m 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. /*
  2. * Licensed under the Apache License, Version 2.0 (the "License");
  3. * you may not use this file except in compliance with the License.
  4. * See the NOTICE file distributed with this work for additional
  5. * information regarding copyright ownership.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import "LRUCache.h"
  17. #import "LRUCacheNode.h"
  18. @interface LRUCache ()
  19. @property (nonatomic) NSMutableDictionary *store;
  20. @property (nonatomic, nullable) LRUCacheNode *headNode;
  21. @property (nonatomic, nullable) LRUCacheNode *tailNode;
  22. @end
  23. @implementation LRUCache
  24. - (instancetype)initWithCapacity:(NSUInteger)capacity
  25. {
  26. if ((self = [super init])) {
  27. _store = [NSMutableDictionary dictionary];
  28. _capacity = capacity;
  29. }
  30. return self;
  31. }
  32. - (void)setObject:(id)object forKey:(id<NSCopying>)key
  33. {
  34. NSAssert(nil != object && nil != key, @"LRUCache cannot store nil objects");
  35. LRUCacheNode *previousNode = self.store[key];
  36. if (nil != previousNode) {
  37. [self removeNode:previousNode];
  38. }
  39. LRUCacheNode *newNode = [LRUCacheNode nodeWithValue:object key:key];
  40. self.store[key] = newNode;
  41. [self addNodeToHead:newNode];
  42. if (nil == previousNode) {
  43. [self alignSize];
  44. }
  45. }
  46. - (id)objectForKey:(id<NSCopying>)key
  47. {
  48. LRUCacheNode *node = self.store[key];
  49. return [self moveNodeToHead:node].value;
  50. }
  51. - (NSArray *)allObjects
  52. {
  53. NSMutableArray *result = [[NSMutableArray alloc] initWithCapacity:self.store.count];
  54. LRUCacheNode *node = self.headNode;
  55. while (node != nil) {
  56. [result addObject:node.value];
  57. node = node.next;
  58. }
  59. return result.copy;
  60. }
  61. - (nullable LRUCacheNode *)moveNodeToHead:(nullable LRUCacheNode *)node
  62. {
  63. if (nil == node || node == self.headNode) {
  64. return node;
  65. }
  66. LRUCacheNode *previousNode = node.prev;
  67. if (nil != previousNode) {
  68. previousNode.next = node.next;
  69. }
  70. LRUCacheNode *nextNode = node.next;
  71. if (nil != nextNode) {
  72. nextNode.prev = node.prev;
  73. }
  74. if (node == self.tailNode) {
  75. self.tailNode = previousNode;
  76. }
  77. return [self addNodeToHead:node];
  78. }
  79. - (void)removeNode:(nullable LRUCacheNode *)node
  80. {
  81. if (nil == node) {
  82. return;
  83. }
  84. if (nil != node.next) {
  85. node.next.prev = node.prev;
  86. }
  87. if (node == self.headNode) {
  88. self.headNode = node.next;
  89. }
  90. if (nil != node.prev) {
  91. node.prev.next = node.next;
  92. }
  93. if (node == self.tailNode) {
  94. self.tailNode = node.prev;
  95. }
  96. [self.store removeObjectForKey:(id)node.key];
  97. }
  98. - (nullable LRUCacheNode *)addNodeToHead:(nullable LRUCacheNode *)node
  99. {
  100. if (nil == node || node == self.headNode) {
  101. return node;
  102. }
  103. LRUCacheNode *previousHead = self.headNode;
  104. if (nil != previousHead) {
  105. previousHead.prev = node;
  106. }
  107. node.next = previousHead;
  108. node.prev = nil;
  109. self.headNode = node;
  110. if (nil == self.tailNode) {
  111. self.tailNode = previousHead ?: node;
  112. }
  113. return node;
  114. }
  115. - (void)alignSize
  116. {
  117. if (self.store.count > self.capacity && nil != self.tailNode) {
  118. [self removeNode:self.tailNode];
  119. }
  120. }
  121. @end