NSCache

一、简介

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 是线程安全的。

你可能感兴趣的:(NSCache)