NSCache

作者: 风轻鱼蛋 | 来源:发表于2017-09-13 14:23 被阅读0次

    一、简介

    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

          本文链接:https://www.haomeiwen.com/subject/apyijxtx.html