一、简介
NSCache
是苹果推出专门用来处理内存缓存的类,是一种类似于 NSDictionary
的缓存。可设置缓存的数目与内存大小限制的方式,默认是线程安全的,在使用的时候可以不用考虑线程安全的问题,当内存警告时内部自动清理部分缓存数据。
1.属性
-
totalCostLimit
设置缓存占用的内存大小,并不是一个严格的限制,当总数超过了totalCostLimit
设定的值,系统会清除一部分缓存,直至总消耗低于totalCostLimit
的值。 -
countLimit
设置缓存对象的大小,这也不是一个严格的限制。
2.方法
//获取缓存对象,基于key-value对
- (id)objectForKey:(id)key;
//存储缓存对象,考虑缓存的限制属性;
- (void)setObject:(id)obj forKey:(id)key; // 0 cost
//存储缓存对象,cost是提前知道该缓存对象占用的字节数,也会考虑缓存的限制属性,建议直接使用 - (void)setObject:(id)obj forKey:(id)key;
- (void)setObject:(id)obj forKey:(id)key cost:(NSUInteger)g;
3.代理
@property (assign) id<NSCacheDelegate>delegate;
实现了NSCacheDelegate代理的对象,在缓存对象即将被清理的时候,系统回调代理方法如下:
/**
1.
第一个参数是当前缓存(NSCache),不要修改该对象;
第二个参数是当前将要被清理的对象,如果需要存储该对象,可以在此操作(存入Sqlite or CoreData);
*/
- (void)cache:(NSCache *)cache willEvictObject:(id)obj;
/**
该代理方法的调用会在缓存对象即将被清理的时候调用,如下场景会调用:
1. - (void)removeObjectForKey:(id)key; 手动删除对象;
2. 缓存对象超过了NSCache的属性限制;(countLimit 和 totalCostLimit )
3. App进入后台会调用;
4. 系统发出内存警告;
*/
4.NSDiscardableContent
协议
NSDiscardableContent
是一个协议,实现这个协议的目的是为了让我们的对象在不被使用时,可以将其丢弃,以让程序占用更少的内存。
一个 NSDiscardableContent
对象的生命周期依赖于一个 counter
变量。一个 NSDiscardableContent
对象实际是一个可清理内存块,这个内存记录了对象当前是否被其它对象使用。如果这块内存正在被读取,或者仍然被需要,则它的 counter
变量是大于或等于 1 的;当它不再被使用时,就可以丢弃,此时 counter
变量将等于 0。当 counter
变量等于 0 时,如果当前时间点内存比较紧张的话,内存块就可能被丢弃。
为了丢弃这些内容,可以调用对象的 discardContentIfPossible
方法
该方法的声明如下:
- (void)discardContentIfPossible;
这样当 counter
变量等于 0 时将会释放相关的内存。而如果 counter
变量不为 0,则该方法什么也不做。
默认情况下,NSDiscardableContent
对象的 counter
变量初始值为 1,以确保对象不会被内存管理系统立即释放。从这个点开始,我们就需要去跟踪 counter
变量的状态。为此。协议声明了两个方法:beginContentAccess
和endContentAccess
。
其中调用 beginContentAccess
方法会增加对象的 counter
变量(+1),这样就可以确保对象不会被丢弃。该方法声明如下:
- (BOOL)beginContentAccess;
通常我们在对象被需要或者将要使用时调用这个方法。具体的实现类可以决定在对象已经被丢弃的情况下是否重新创建这些内存,且重新创建成功后返回 YES。协议的实现者在 NSDiscardableContent
对象被使用,而又没有调用它的 beginContentAccess
方法时,应该抛出一个异常。
函数的返回值如果是 YES,则表明可丢弃内存仍然可用且已被成功访问;否则返回 NO。另外需要注意的是,该方法是在实现类中必须实现(required)。
与beginContentAccess
相对应的是 endContentAccess
。如果可丢弃内存不再被访问时调用。其声明如下:
- (void)endContentAccess;
该方法会减少对象的 counter
变量,通常是让对象的 counter
值变回为 0,这样在对象的内容不再被需要时,就要以将其丢弃。
NSCache
类提供了一个属性,来标识缓存是否自动舍弃那些内存已经被丢弃的对象(discardable-content object),其声明如下:
@property BOOL evictsObjectsWithDiscardedContent;
如果设置为 YES,则在对象的内存被丢弃时舍弃对象。默认值为 YES。
二、使用
1.缓存数量限制
#import "ViewController.h"
@interface ViewController () <NSCacheDelegate>
@property (nonatomic, strong) NSCache *cache;
@end
@implementation ViewController
//实现代理方法:
// MARK: NSCache Delegate
// 当缓存中的对象被清除的时候,会自动调用
// obj 就是要被清理的对象
// 提示:不建议平时开发时重写!仅供调试使用
- (void)cache:(NSCache *)cache willEvictObject:(id)obj {
[NSThread sleepForTimeInterval:0.5];
NSLog(@"清除了-------> %@", obj);
}
- (NSCache *)cache {
if (_cache == nil) {
_cache = [[NSCache alloc] init];
// 设置数量限制,最大限制为10
_cache.countLimit = 3;
_cache.delegate = self;
}
return _cache;
}
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
for (int i = 0; i < 6; ++i) {
NSString *str = [NSString stringWithFormat:@"hello - %04d", i];
NSLog(@"设置 %@", str);
// 添加到缓存
[self.cache setObject:str forKey:@(i)];
}
// - 查看缓存内容,NSCache 没有提供遍历的方法,只支持用 key 来取值
for (int i = 0; i < 6; ++i) {
NSLog(@"缓存中----->%@", [self.cache objectForKey:@(i)]);
}
}
@end
2022-07-29 16:43:32.310988+0800 DJTestDemo[8324:4524183] 设置 hello - 0000
2022-07-29 16:43:32.311344+0800 DJTestDemo[8324:4524183] 设置 hello - 0001
2022-07-29 16:43:32.311498+0800 DJTestDemo[8324:4524183] 设置 hello - 0002
2022-07-29 16:43:32.311686+0800 DJTestDemo[8324:4524183] 设置 hello - 0003
2022-07-29 16:43:32.813408+0800 DJTestDemo[8324:4524183] 清除了-------> hello - 0000
2022-07-29 16:43:32.813831+0800 DJTestDemo[8324:4524183] 设置 hello - 0004
2022-07-29 16:43:33.315204+0800 DJTestDemo[8324:4524183] 清除了-------> hello - 0001
2022-07-29 16:43:33.315620+0800 DJTestDemo[8324:4524183] 设置 hello - 0005
2022-07-29 16:43:33.815907+0800 DJTestDemo[8324:4524183] 清除了-------> hello - 0002
2022-07-29 16:43:33.816244+0800 DJTestDemo[8324:4524183] 缓存中----->(null)
2022-07-29 16:43:33.816501+0800 DJTestDemo[8324:4524183] 缓存中----->(null)
2022-07-29 16:43:33.816727+0800 DJTestDemo[8324:4524183] 缓存中----->(null)
2022-07-29 16:43:33.816921+0800 DJTestDemo[8324:4524183] 缓存中----->hello - 0003
2022-07-29 16:43:33.817109+0800 DJTestDemo[8324:4524183] 缓存中----->hello - 0004
2022-07-29 16:43:33.817286+0800 DJTestDemo[8324:4524183] 缓存中----->hello - 0005