一、简介
1、NSCache是苹果官方提供的的缓存类,具体使用是和NSDictionary很相似的,在AFNetworking和SDWebImage第三方框架中被用作管理缓存。
2、NSCache在系统内存很低时会自动释放对象(在模拟器进行演示的时候不会释放,这一点要注意,所以最好是在内存进行警告时主动去调用方法去释放对象)。
3、NSCache是线程安全的,在进行多线程操作时,不需要进行加锁。
4、NSCache的Key只是对对象进行了Strong引用,而非拷贝,相关说明会在下面的例子中进行详细解释。
属性和方法
@property NSUInteger totalCostLimit; //缓存空间的最大成本,超出上限会自动回收对象,默认值是0
@property NSUInteger countLimit; //能够缓存对象的最大数量,默认值是 0
@property BOOL evictsObjectsWithDiscardedContent; // 标识缓存是否回收废弃的内容, 默认YES
// 通过指定的键取出在缓存中存储的数据。
- (nullable ObjectType)objectForKey:(KeyType)key;
// 在缓存中设置指定键名对应的值,成本为0。
- (void)setObject:(ObjectType)obj forKey:(KeyType)key;
// 在缓存中设置指定键名对应的值,并且指定回收成本,以便进行计算存储在缓存中对象的总成本,当出现内存警告或者超出总成本时,缓存就会进行删除部分元素的操作。
- (void)setObject:(ObjectType)obj forKey:(KeyType)key cost:(NSUInteger)g;
// 通过指定的键清除在缓存中存储的数据。
- (void)removeObjectForKey:(KeyType)key;
// 清除在缓存中存储的所有数据。
- (void)removeAllObjects;
三、简单使用
1、创建缓存对象
- (NSCache *)cache
{
if (!_cache) {
_cache = [[NSCache alloc]init];
// 设置成本为5 当存储的数据超过总成本数,NSCache会自动回收对象
_cache.totalCostLimit = 5;
// 设置代理 代理方法一般不会用到,一般是进行测试的时候使用
_cache.delegate = self;
}
return _cache;
}
2、缓存操作
//添加缓存
- (IBAction)addCache:(id)sender {
for (int i = 0; i < 10; i ++) {
NSString *cacheStr = [NSString stringWithFormat:@"缓存数据%zd",i];
[self.cache setObject:cacheStr forKey:@(i) cost:1];
NSLog(@"储存数据---%@---",cacheStr);
}
}
//读取缓存
- (IBAction)readCache:(id)sender {
for (int i = 0; i < 10; i ++) {
NSString *str = [self.cache objectForKey:@(i)];
if (str.length) {
NSLog(@"读取数据---%@---",str);
}
}
}
//删除缓存
- (IBAction)deleteCache:(id)sender {
[self.cache removeAllObjects];
NSLog(@"删除了所有数据");
}
#pragma mark - NSCacheDelegate
- (void)cache:(NSCache *)cache willEvictObject:(id)obj
{
NSLog(@"回收----%@----",obj);
}
3、打印结果
3.1 点击添加按钮的打印
2017-09-13 13:57:56.297 TextDemo[1589:479484] 储存数据---缓存数据2---
2017-09-13 13:57:56.297 TextDemo[1589:479484] 储存数据---缓存数据3---
2017-09-13 13:57:56.297 TextDemo[1589:479484] 储存数据---缓存数据4---
2017-09-13 13:57:56.297 TextDemo[1589:479484] 回收----缓存数据0----
2017-09-13 13:57:56.297 TextDemo[1589:479484] 储存数据---缓存数据5---
2017-09-13 13:57:56.298 TextDemo[1589:479484] 回收----缓存数据1----
2017-09-13 13:57:56.298 TextDemo[1589:479484] 储存数据---缓存数据6---
2017-09-13 13:57:56.298 TextDemo[1589:479484] 回收----缓存数据2----
2017-09-13 13:57:56.298 TextDemo[1589:479484] 储存数据---缓存数据7---
2017-09-13 13:57:56.298 TextDemo[1589:479484] 回收----缓存数据3----
2017-09-13 13:57:56.298 TextDemo[1589:479484] 储存数据---缓存数据8---
2017-09-13 13:57:56.298 TextDemo[1589:479484] 回收----缓存数据4----
2017-09-13 13:57:56.299 TextDemo[1589:479484] 储存数据---缓存数据9---
进行每个字符串对象存储时,成本是1,我们设置的总成本是5,字符串对象存储了10次,总成本是10,所以在存储数据5的时候会回收数据1的字符串对象,以此类推,所以打印的结果如上所示。
3.2 点击读取按钮的打印
2017-09-13 13:58:40.700 TextDemo[1589:479484] 读取数据---缓存数据5---
2017-09-13 13:58:40.700 TextDemo[1589:479484] 读取数据---缓存数据6---
2017-09-13 13:58:40.700 TextDemo[1589:479484] 读取数据---缓存数据7---
2017-09-13 13:58:40.700 TextDemo[1589:479484] 读取数据---缓存数据8---
2017-09-13 13:58:40.700 TextDemo[1589:479484] 读取数据---缓存数据9---
3.3 点击删除按钮的打印
2017-09-13 13:58:54.290 TextDemo[1589:479484] 回收----缓存数据5----
2017-09-13 13:58:54.290 TextDemo[1589:479484] 回收----缓存数据6----
2017-09-13 13:58:54.290 TextDemo[1589:479484] 回收----缓存数据7----
2017-09-13 13:58:54.291 TextDemo[1589:479484] 回收----缓存数据8----
2017-09-13 13:58:54.291 TextDemo[1589:479484] 回收----缓存数据9----
2017-09-13 13:58:54.291 TextDemo[1589:479484] 删除了所有数据
四、补充说明
1、NSCache的Key只是对对象进行了Strong引用,而非拷贝。
当cacheStr
创建写在for循环外部时,存储的真正的字符串对象只有一个,对字符串对象只是在内存中建立了10个强引用,所以总成本为1。
当cacheStr
创建写在for循环内部时,存储的真正的字符串对象有十个,所以总成本为10。
2、NSCache中的数据在APP重启(或重新init)后,都消失了,并不会保存下来。NSCache只是将数据保存在内存中。
3、释放内存时,并不确定释放的对象的顺序。
4、NSCache 是线程安全的。