123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694 |
- // PINCache is a modified version of TMCache
- // Modifications by Garrett Moon
- // Copyright (c) 2015 Pinterest. All rights reserved.
- #import "PINMemoryCache.h"
- #if __IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0
- #import <UIKit/UIKit.h>
- #endif
- static NSString * const PINMemoryCachePrefix = @"com.pinterest.PINMemoryCache";
- @interface PINMemoryCache ()
- #if OS_OBJECT_USE_OBJC
- @property (strong, nonatomic) dispatch_queue_t concurrentQueue;
- @property (strong, nonatomic) dispatch_semaphore_t lockSemaphore;
- #else
- @property (assign, nonatomic) dispatch_queue_t concurrentQueue;
- @property (assign, nonatomic) dispatch_semaphore_t lockSemaphore;
- #endif
- @property (strong, nonatomic) NSMutableDictionary *dictionary;
- @property (strong, nonatomic) NSMutableDictionary *dates;
- @property (strong, nonatomic) NSMutableDictionary *costs;
- @end
- @implementation PINMemoryCache
- @synthesize ageLimit = _ageLimit;
- @synthesize costLimit = _costLimit;
- @synthesize totalCost = _totalCost;
- @synthesize willAddObjectBlock = _willAddObjectBlock;
- @synthesize willRemoveObjectBlock = _willRemoveObjectBlock;
- @synthesize willRemoveAllObjectsBlock = _willRemoveAllObjectsBlock;
- @synthesize didAddObjectBlock = _didAddObjectBlock;
- @synthesize didRemoveObjectBlock = _didRemoveObjectBlock;
- @synthesize didRemoveAllObjectsBlock = _didRemoveAllObjectsBlock;
- @synthesize didReceiveMemoryWarningBlock = _didReceiveMemoryWarningBlock;
- @synthesize didEnterBackgroundBlock = _didEnterBackgroundBlock;
- #pragma mark - Initialization -
- - (void)dealloc
- {
- [[NSNotificationCenter defaultCenter] removeObserver:self];
- #if !OS_OBJECT_USE_OBJC
- dispatch_release(_concurrentQueue);
- dispatch_release(_lockSemaphore);
- _concurrentQueue = nil;
- #endif
- }
- - (instancetype)init
- {
- if (self = [super init]) {
- _lockSemaphore = dispatch_semaphore_create(1);
- NSString *queueName = [[NSString alloc] initWithFormat:@"%@.%p", PINMemoryCachePrefix, self];
- _concurrentQueue = dispatch_queue_create([queueName UTF8String], DISPATCH_QUEUE_CONCURRENT);
- _dictionary = [[NSMutableDictionary alloc] init];
- _dates = [[NSMutableDictionary alloc] init];
- _costs = [[NSMutableDictionary alloc] init];
- _willAddObjectBlock = nil;
- _willRemoveObjectBlock = nil;
- _willRemoveAllObjectsBlock = nil;
- _didAddObjectBlock = nil;
- _didRemoveObjectBlock = nil;
- _didRemoveAllObjectsBlock = nil;
- _didReceiveMemoryWarningBlock = nil;
- _didEnterBackgroundBlock = nil;
- _ageLimit = 0.0;
- _costLimit = 0;
- _totalCost = 0;
- _removeAllObjectsOnMemoryWarning = YES;
- _removeAllObjectsOnEnteringBackground = YES;
- #if TARGET_OS_IPHONE && defined(__IPHONE_OS_VERSION_MIN_REQUIRED) && __IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0 && !TARGET_OS_WATCH
- for (NSString *name in @[UIApplicationDidReceiveMemoryWarningNotification, UIApplicationDidEnterBackgroundNotification]) {
- [[NSNotificationCenter defaultCenter] addObserver:self
- selector:@selector(didObserveApocalypticNotification:)
- name:name
- #if !defined(PIN_APP_EXTENSIONS)
- object:[UIApplication sharedApplication]];
- #else
- object:nil];
- #endif
- }
- #endif
- }
- return self;
- }
- + (instancetype)sharedCache
- {
- static id cache;
- static dispatch_once_t predicate;
- dispatch_once(&predicate, ^{
- cache = [[self alloc] init];
- });
- return cache;
- }
- #pragma mark - Private Methods -
- - (void)didObserveApocalypticNotification:(NSNotification *)notification
- {
- #if TARGET_OS_IPHONE && defined(__IPHONE_OS_VERSION_MIN_REQUIRED) && __IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_4_0 && !TARGET_OS_WATCH
- if ([[notification name] isEqualToString:UIApplicationDidReceiveMemoryWarningNotification]) {
- if (self.removeAllObjectsOnMemoryWarning)
- [self removeAllObjects:nil];
- __weak PINMemoryCache *weakSelf = self;
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- if (!strongSelf) {
- return;
- }
-
- [strongSelf lock];
- PINMemoryCacheBlock didReceiveMemoryWarningBlock = strongSelf->_didReceiveMemoryWarningBlock;
- [strongSelf unlock];
-
- if (didReceiveMemoryWarningBlock)
- didReceiveMemoryWarningBlock(strongSelf);
- });
- } else if ([[notification name] isEqualToString:UIApplicationDidEnterBackgroundNotification]) {
- if (self.removeAllObjectsOnEnteringBackground)
- [self removeAllObjects:nil];
- __weak PINMemoryCache *weakSelf = self;
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- if (!strongSelf) {
- return;
- }
- [strongSelf lock];
- PINMemoryCacheBlock didEnterBackgroundBlock = strongSelf->_didEnterBackgroundBlock;
- [strongSelf unlock];
-
- if (didEnterBackgroundBlock)
- didEnterBackgroundBlock(strongSelf);
- });
- }
-
- #endif
- }
- - (void)removeObjectAndExecuteBlocksForKey:(NSString *)key
- {
- [self lock];
- id object = _dictionary[key];
- NSNumber *cost = _costs[key];
- PINMemoryCacheObjectBlock willRemoveObjectBlock = _willRemoveObjectBlock;
- PINMemoryCacheObjectBlock didRemoveObjectBlock = _didRemoveObjectBlock;
- [self unlock];
- if (willRemoveObjectBlock)
- willRemoveObjectBlock(self, key, object);
- [self lock];
- if (cost)
- _totalCost -= [cost unsignedIntegerValue];
- [_dictionary removeObjectForKey:key];
- [_dates removeObjectForKey:key];
- [_costs removeObjectForKey:key];
- [self unlock];
-
- if (didRemoveObjectBlock)
- didRemoveObjectBlock(self, key, nil);
- }
- - (void)trimMemoryToDate:(NSDate *)trimDate
- {
- [self lock];
- NSArray *keysSortedByDate = [_dates keysSortedByValueUsingSelector:@selector(compare:)];
- NSDictionary *dates = [_dates copy];
- [self unlock];
-
- for (NSString *key in keysSortedByDate) { // oldest objects first
- NSDate *accessDate = dates[key];
- if (!accessDate)
- continue;
-
- if ([accessDate compare:trimDate] == NSOrderedAscending) { // older than trim date
- [self removeObjectAndExecuteBlocksForKey:key];
- } else {
- break;
- }
- }
- }
- - (void)trimToCostLimit:(NSUInteger)limit
- {
- NSUInteger totalCost = 0;
-
- [self lock];
- totalCost = _totalCost;
- NSArray *keysSortedByCost = [_costs keysSortedByValueUsingSelector:@selector(compare:)];
- [self unlock];
-
- if (totalCost <= limit) {
- return;
- }
- for (NSString *key in [keysSortedByCost reverseObjectEnumerator]) { // costliest objects first
- [self removeObjectAndExecuteBlocksForKey:key];
- [self lock];
- totalCost = _totalCost;
- [self unlock];
-
- if (totalCost <= limit)
- break;
- }
- }
- - (void)trimToCostLimitByDate:(NSUInteger)limit
- {
- NSUInteger totalCost = 0;
-
- [self lock];
- totalCost = _totalCost;
- NSArray *keysSortedByDate = [_dates keysSortedByValueUsingSelector:@selector(compare:)];
- [self unlock];
-
- if (totalCost <= limit)
- return;
- for (NSString *key in keysSortedByDate) { // oldest objects first
- [self removeObjectAndExecuteBlocksForKey:key];
- [self lock];
- totalCost = _totalCost;
- [self unlock];
- if (totalCost <= limit)
- break;
- }
- }
- - (void)trimToAgeLimitRecursively
- {
- [self lock];
- NSTimeInterval ageLimit = _ageLimit;
- [self unlock];
-
- if (ageLimit == 0.0)
- return;
- NSDate *date = [[NSDate alloc] initWithTimeIntervalSinceNow:-ageLimit];
-
- [self trimMemoryToDate:date];
-
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_time_t time = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(ageLimit * NSEC_PER_SEC));
- dispatch_after(time, _concurrentQueue, ^(void){
- PINMemoryCache *strongSelf = weakSelf;
-
- [strongSelf trimToAgeLimitRecursively];
- });
- }
- #pragma mark - Public Asynchronous Methods -
- - (void)objectForKey:(NSString *)key block:(PINMemoryCacheObjectBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- id object = [strongSelf objectForKey:key];
-
- if (block)
- block(strongSelf, key, object);
- });
- }
- - (void)setObject:(id)object forKey:(NSString *)key block:(PINMemoryCacheObjectBlock)block
- {
- [self setObject:object forKey:key withCost:0 block:block];
- }
- - (void)setObject:(id)object forKey:(NSString *)key withCost:(NSUInteger)cost block:(PINMemoryCacheObjectBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf setObject:object forKey:key withCost:cost];
-
- if (block)
- block(strongSelf, key, object);
- });
- }
- - (void)removeObjectForKey:(NSString *)key block:(PINMemoryCacheObjectBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf removeObjectForKey:key];
-
- if (block)
- block(strongSelf, key, nil);
- });
- }
- - (void)trimToDate:(NSDate *)trimDate block:(PINMemoryCacheBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf trimToDate:trimDate];
-
- if (block)
- block(strongSelf);
- });
- }
- - (void)trimToCost:(NSUInteger)cost block:(PINMemoryCacheBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf trimToCost:cost];
-
- if (block)
- block(strongSelf);
- });
- }
- - (void)trimToCostByDate:(NSUInteger)cost block:(PINMemoryCacheBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf trimToCostByDate:cost];
-
- if (block)
- block(strongSelf);
- });
- }
- - (void)removeAllObjects:(PINMemoryCacheBlock)block
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf removeAllObjects];
-
- if (block)
- block(strongSelf);
- });
- }
- - (void)enumerateObjectsWithBlock:(PINMemoryCacheObjectBlock)block completionBlock:(PINMemoryCacheBlock)completionBlock
- {
- __weak PINMemoryCache *weakSelf = self;
-
- dispatch_async(_concurrentQueue, ^{
- PINMemoryCache *strongSelf = weakSelf;
- [strongSelf enumerateObjectsWithBlock:block];
-
- if (completionBlock)
- completionBlock(strongSelf);
- });
- }
- #pragma mark - Public Synchronous Methods -
- - (__nullable id)objectForKey:(NSString *)key
- {
- if (!key)
- return nil;
-
- [self lock];
- id object = _dictionary[key];
- [self unlock];
-
- if (object) {
- [self lock];
- _dates[key] = [[NSDate alloc] init];
- [self unlock];
- }
- return object;
- }
- - (void)setObject:(id)object forKey:(NSString *)key
- {
- [self setObject:object forKey:key withCost:0];
- }
- - (void)setObject:(id)object forKey:(NSString *)key withCost:(NSUInteger)cost
- {
- if (!key || !object)
- return;
-
- [self lock];
- PINMemoryCacheObjectBlock willAddObjectBlock = _willAddObjectBlock;
- PINMemoryCacheObjectBlock didAddObjectBlock = _didAddObjectBlock;
- NSUInteger costLimit = _costLimit;
- [self unlock];
-
- if (willAddObjectBlock)
- willAddObjectBlock(self, key, object);
-
- [self lock];
- _dictionary[key] = object;
- _dates[key] = [[NSDate alloc] init];
- _costs[key] = @(cost);
-
- _totalCost += cost;
- [self unlock];
-
- if (didAddObjectBlock)
- didAddObjectBlock(self, key, object);
-
- if (costLimit > 0)
- [self trimToCostByDate:costLimit];
- }
- - (void)removeObjectForKey:(NSString *)key
- {
- if (!key)
- return;
-
- [self removeObjectAndExecuteBlocksForKey:key];
- }
- - (void)trimToDate:(NSDate *)trimDate
- {
- if (!trimDate)
- return;
-
- if ([trimDate isEqualToDate:[NSDate distantPast]]) {
- [self removeAllObjects];
- return;
- }
-
- [self trimMemoryToDate:trimDate];
- }
- - (void)trimToCost:(NSUInteger)cost
- {
- [self trimToCostLimit:cost];
- }
- - (void)trimToCostByDate:(NSUInteger)cost
- {
- [self trimToCostLimitByDate:cost];
- }
- - (void)removeAllObjects
- {
- [self lock];
- PINMemoryCacheBlock willRemoveAllObjectsBlock = _willRemoveAllObjectsBlock;
- PINMemoryCacheBlock didRemoveAllObjectsBlock = _didRemoveAllObjectsBlock;
- [self unlock];
-
- if (willRemoveAllObjectsBlock)
- willRemoveAllObjectsBlock(self);
-
- [self lock];
- [_dictionary removeAllObjects];
- [_dates removeAllObjects];
- [_costs removeAllObjects];
-
- _totalCost = 0;
- [self unlock];
-
- if (didRemoveAllObjectsBlock)
- didRemoveAllObjectsBlock(self);
-
- }
- - (void)enumerateObjectsWithBlock:(PINMemoryCacheObjectBlock)block
- {
- if (!block)
- return;
-
- [self lock];
- NSArray *keysSortedByDate = [_dates keysSortedByValueUsingSelector:@selector(compare:)];
-
- for (NSString *key in keysSortedByDate) {
- block(self, key, _dictionary[key]);
- }
- [self unlock];
- }
- #pragma mark - Public Thread Safe Accessors -
- - (PINMemoryCacheObjectBlock)willAddObjectBlock
- {
- [self lock];
- PINMemoryCacheObjectBlock block = _willAddObjectBlock;
- [self unlock];
- return block;
- }
- - (void)setWillAddObjectBlock:(PINMemoryCacheObjectBlock)block
- {
- [self lock];
- _willAddObjectBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheObjectBlock)willRemoveObjectBlock
- {
- [self lock];
- PINMemoryCacheObjectBlock block = _willRemoveObjectBlock;
- [self unlock];
- return block;
- }
- - (void)setWillRemoveObjectBlock:(PINMemoryCacheObjectBlock)block
- {
- [self lock];
- _willRemoveObjectBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheBlock)willRemoveAllObjectsBlock
- {
- [self lock];
- PINMemoryCacheBlock block = _willRemoveAllObjectsBlock;
- [self unlock];
- return block;
- }
- - (void)setWillRemoveAllObjectsBlock:(PINMemoryCacheBlock)block
- {
- [self lock];
- _willRemoveAllObjectsBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheObjectBlock)didAddObjectBlock
- {
- [self lock];
- PINMemoryCacheObjectBlock block = _didAddObjectBlock;
- [self unlock];
- return block;
- }
- - (void)setDidAddObjectBlock:(PINMemoryCacheObjectBlock)block
- {
- [self lock];
- _didAddObjectBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheObjectBlock)didRemoveObjectBlock
- {
- [self lock];
- PINMemoryCacheObjectBlock block = _didRemoveObjectBlock;
- [self unlock];
- return block;
- }
- - (void)setDidRemoveObjectBlock:(PINMemoryCacheObjectBlock)block
- {
- [self lock];
- _didRemoveObjectBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheBlock)didRemoveAllObjectsBlock
- {
- [self lock];
- PINMemoryCacheBlock block = _didRemoveAllObjectsBlock;
- [self unlock];
- return block;
- }
- - (void)setDidRemoveAllObjectsBlock:(PINMemoryCacheBlock)block
- {
- [self lock];
- _didRemoveAllObjectsBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheBlock)didReceiveMemoryWarningBlock
- {
- [self lock];
- PINMemoryCacheBlock block = _didReceiveMemoryWarningBlock;
- [self unlock];
- return block;
- }
- - (void)setDidReceiveMemoryWarningBlock:(PINMemoryCacheBlock)block
- {
- [self lock];
- _didReceiveMemoryWarningBlock = [block copy];
- [self unlock];
- }
- - (PINMemoryCacheBlock)didEnterBackgroundBlock
- {
- [self lock];
- PINMemoryCacheBlock block = _didEnterBackgroundBlock;
- [self unlock];
- return block;
- }
- - (void)setDidEnterBackgroundBlock:(PINMemoryCacheBlock)block
- {
- [self lock];
- _didEnterBackgroundBlock = [block copy];
- [self unlock];
- }
- - (NSTimeInterval)ageLimit
- {
- [self lock];
- NSTimeInterval ageLimit = _ageLimit;
- [self unlock];
-
- return ageLimit;
- }
- - (void)setAgeLimit:(NSTimeInterval)ageLimit
- {
- [self lock];
- _ageLimit = ageLimit;
- [self unlock];
-
- [self trimToAgeLimitRecursively];
- }
- - (NSUInteger)costLimit
- {
- [self lock];
- NSUInteger costLimit = _costLimit;
- [self unlock];
- return costLimit;
- }
- - (void)setCostLimit:(NSUInteger)costLimit
- {
- [self lock];
- _costLimit = costLimit;
- [self unlock];
- if (costLimit > 0)
- [self trimToCostLimitByDate:costLimit];
- }
- - (NSUInteger)totalCost
- {
- [self lock];
- NSUInteger cost = _totalCost;
- [self unlock];
-
- return cost;
- }
- - (void)lock
- {
- dispatch_semaphore_wait(_lockSemaphore, DISPATCH_TIME_FOREVER);
- }
- - (void)unlock
- {
- dispatch_semaphore_signal(_lockSemaphore);
- }
- @end
|