/** `TMMemoryCache` is a fast, thread safe key/value store similar to `NSCache`. On iOS it will clear itself automatically to reduce memory usage when the app receives a memory warning or goes into the background. Access is natively asynchronous. Every method accepts a callback block that runs on a concurrent , with cache writes protected by GCD barriers. Synchronous variations are provided. All access to the cache is dated so the that the least-used objects can be trimmed first. Setting an optional will trigger a GCD timer to periodically to trim the cache to that age. Objects can optionally be set with a "cost", which could be a byte count or any other meaningful integer. Setting a will automatically keep the cache below that value with . Values will not persist after application relaunch or returning from the background. See for a memory cache backed by a disk cache. */ #import @class TMMemoryCache; typedef void (^TMMemoryCacheBlock)(TMMemoryCache *cache); typedef void (^TMMemoryCacheObjectBlock)(TMMemoryCache *cache, NSString *key, id object); @interface TMMemoryCache : NSObject #pragma mark - /// @name Core /** A concurrent queue on which all work is done. It is exposed here so that it can be set to target some other queue, such as a global concurrent queue with a priority other than the default. */ @property (readonly) dispatch_queue_t queue; /** The total accumulated cost. */ @property (readonly) NSUInteger totalCost; /** The maximum cost allowed to accumulate before objects begin to be removed with . */ @property (assign) NSUInteger costLimit; /** The maximum number of seconds an object is allowed to exist in the cache. Setting this to a value greater than `0.0` will start a recurring GCD timer with the same period that calls . Setting it back to `0.0` will stop the timer. Defaults to `0.0`. */ @property (assign) NSTimeInterval ageLimit; /** When `YES` on iOS the cache will remove all objects when the app receives a memory warning. Defaults to `YES`. */ @property (assign) BOOL removeAllObjectsOnMemoryWarning; /** When `YES` on iOS the cache will remove all objects when the app enters the background. Defaults to `YES`. */ @property (assign) BOOL removeAllObjectsOnEnteringBackground; #pragma mark - /// @name Event Blocks /** A block to be executed just before an object is added to the cache. This block will be excuted within a barrier, i.e. all reads and writes are suspended for the duration of the block. */ @property (copy) TMMemoryCacheObjectBlock willAddObjectBlock; /** A block to be executed just before an object is removed from the cache. This block will be excuted within a barrier, i.e. all reads and writes are suspended for the duration of the block. */ @property (copy) TMMemoryCacheObjectBlock willRemoveObjectBlock; /** A block to be executed just before all objects are removed from the cache as a result of . This block will be excuted within a barrier, i.e. all reads and writes are suspended for the duration of the block. */ @property (copy) TMMemoryCacheBlock willRemoveAllObjectsBlock; /** A block to be executed just after an object is added to the cache. This block will be excuted within a barrier, i.e. all reads and writes are suspended for the duration of the block. */ @property (copy) TMMemoryCacheObjectBlock didAddObjectBlock; /** A block to be executed just after an object is removed from the cache. This block will be excuted within a barrier, i.e. all reads and writes are suspended for the duration of the block. */ @property (copy) TMMemoryCacheObjectBlock didRemoveObjectBlock; /** A block to be executed just after all objects are removed from the cache as a result of . This block will be excuted within a barrier, i.e. all reads and writes are suspended for the duration of the block. */ @property (copy) TMMemoryCacheBlock didRemoveAllObjectsBlock; /** A block to be executed upon receiving a memory warning (iOS only) potentially in parallel with other blocks on the . This block will be executed regardless of the value of . Defaults to `nil`. */ @property (copy) TMMemoryCacheBlock didReceiveMemoryWarningBlock; /** A block to be executed when the app enters the background (iOS only) potentially in parallel with other blocks on the . This block will be executed regardless of the value of . Defaults to `nil`. */ @property (copy) TMMemoryCacheBlock didEnterBackgroundBlock; #pragma mark - /// @name Shared Cache /** A shared cache. @result The shared singleton cache instance. */ + (instancetype)sharedCache; #pragma mark - /// @name Asynchronous Methods /** Retrieves the object for the specified key. This method returns immediately and executes the passed block after the object is available, potentially in parallel with other blocks on the . @param key The key associated with the requested object. @param block A block to be executed concurrently when the object is available. */ - (void)objectForKey:(NSString *)key block:(TMMemoryCacheObjectBlock)block; /** Stores an object in the cache for the specified key. This method returns immediately and executes the passed block after the object has been stored, potentially in parallel with other blocks on the . @param object An object to store in the cache. @param key A key to associate with the object. This string will be copied. @param block A block to be executed concurrently after the object has been stored, or nil. */ - (void)setObject:(id)object forKey:(NSString *)key block:(TMMemoryCacheObjectBlock)block; /** Stores an object in the cache for the specified key and the specified cost. If the cost causes the total to go over the the cache is trimmed (oldest objects first). This method returns immediately and executes the passed block after the object has been stored, potentially in parallel with other blocks on the . @param object An object to store in the cache. @param key A key to associate with the object. This string will be copied. @param cost An amount to add to the . @param block A block to be executed concurrently after the object has been stored, or nil. */ - (void)setObject:(id)object forKey:(NSString *)key withCost:(NSUInteger)cost block:(TMMemoryCacheObjectBlock)block; /** Removes the object for the specified key. This method returns immediately and executes the passed block after the object has been removed, potentially in parallel with other blocks on the . @param key The key associated with the object to be removed. @param block A block to be executed concurrently after the object has been removed, or nil. */ - (void)removeObjectForKey:(NSString *)key block:(TMMemoryCacheObjectBlock)block; /** Removes all objects from the cache that have not been used since the specified date. This method returns immediately and executes the passed block after the cache has been trimmed, potentially in parallel with other blocks on the . @param date Objects that haven't been accessed since this date are removed from the cache. @param block A block to be executed concurrently after the cache has been trimmed, or nil. */ - (void)trimToDate:(NSDate *)date block:(TMMemoryCacheBlock)block; /** Removes objects from the cache, costliest objects first, until the is below the specified value. This method returns immediately and executes the passed block after the cache has been trimmed, potentially in parallel with other blocks on the . @param cost The total accumulation allowed to remain after the cache has been trimmed. @param block A block to be executed concurrently after the cache has been trimmed, or nil. */ - (void)trimToCost:(NSUInteger)cost block:(TMMemoryCacheBlock)block; /** Removes objects from the cache, ordered by date (least recently used first), until the is below the specified value. This method returns immediately and executes the passed block after the cache has been trimmed, potentially in parallel with other blocks on the . @param cost The total accumulation allowed to remain after the cache has been trimmed. @param block A block to be executed concurrently after the cache has been trimmed, or nil. */ - (void)trimToCostByDate:(NSUInteger)cost block:(TMMemoryCacheBlock)block; /** Removes all objects from the cache. This method returns immediately and executes the passed block after the cache has been cleared, potentially in parallel with other blocks on the . @param block A block to be executed concurrently after the cache has been cleared, or nil. */ - (void)removeAllObjects:(TMMemoryCacheBlock)block; /** Loops through all objects in the cache within a memory barrier (reads and writes are suspended during the enumeration). This method returns immediately. @param block A block to be executed for every object in the cache. @param completionBlock An optional block to be executed concurrently when the enumeration is complete. */ - (void)enumerateObjectsWithBlock:(TMMemoryCacheObjectBlock)block completionBlock:(TMMemoryCacheBlock)completionBlock; #pragma mark - /// @name Synchronous Methods /** Retrieves the object for the specified key. This method blocks the calling thread until the object is available. @see objectForKey:block: @param key The key associated with the object. @result The object for the specified key. */ - (id)objectForKey:(NSString *)key; /** Stores an object in the cache for the specified key. This method blocks the calling thread until the object has been set. @see setObject:forKey:block: @param object An object to store in the cache. @param key A key to associate with the object. This string will be copied. */ - (void)setObject:(id)object forKey:(NSString *)key; /** Stores an object in the cache for the specified key and the specified cost. If the cost causes the total to go over the the cache is trimmed (oldest objects first). This method blocks the calling thread until the object has been stored. @param object An object to store in the cache. @param key A key to associate with the object. This string will be copied. @param cost An amount to add to the . */ - (void)setObject:(id)object forKey:(NSString *)key withCost:(NSUInteger)cost; /** Removes the object for the specified key. This method blocks the calling thread until the object has been removed. @param key The key associated with the object to be removed. */ - (void)removeObjectForKey:(NSString *)key; /** Removes all objects from the cache that have not been used since the specified date. This method blocks the calling thread until the cache has been trimmed. @param date Objects that haven't been accessed since this date are removed from the cache. */ - (void)trimToDate:(NSDate *)date; /** Removes objects from the cache, costliest objects first, until the is below the specified value. This method blocks the calling thread until the cache has been trimmed. @param cost The total accumulation allowed to remain after the cache has been trimmed. */ - (void)trimToCost:(NSUInteger)cost; /** Removes objects from the cache, ordered by date (least recently used first), until the is below the specified value. This method blocks the calling thread until the cache has been trimmed. @param cost The total accumulation allowed to remain after the cache has been trimmed. */ - (void)trimToCostByDate:(NSUInteger)cost; /** Removes all objects from the cache. This method blocks the calling thread until the cache has been cleared. */ - (void)removeAllObjects; /** Loops through all objects in the cache within a memory barrier (reads and writes are suspended during the enumeration). This method blocks the calling thread until all objects have been enumerated. @param block A block to be executed for every object in the cache. @warning Do not call this method within the event blocks (, etc.) Instead use the asynchronous version, . */ - (void)enumerateObjectsWithBlock:(TMMemoryCacheObjectBlock)block; /** Handle a memory warning. */ - (void)handleMemoryWarning __deprecated_msg("This happens automatically in TMCache 2.1. There’s no longer a need to call it directly."); /** Handle the application having been backgrounded. */ - (void)handleApplicationBackgrounding __deprecated_msg("This happens automatically in TMCache 2.1. There’s no longer a need to call it directly."); @end