【问题标题】:NSCache is not evicting dataNSCache 不驱逐数据
【发布时间】:2012-07-16 06:36:04
【问题描述】:

NSCache 是一个很少使用的工具,实际上看起来非常有用。我创建了一个简单的实验来看看它是如何工作的,看起来它不会在内存不足的情况下自动驱逐数据(或者我做错了什么!)

- (void)viewDidLoad
{
    _testCache = [[NSCache alloc] init];

    // Allocate 600 MB of zeros and save to NSCache
    NSMutableData* largeData = [[NSMutableData alloc] init];
    [largeData setLength:1024 * 1024 * 600]; 
    [_testCache setObject:largeData forKey:@"original_Data"];
}

- (IBAction)buttonWasTapped:(id)sender {

    // Allocate & save to cache 300 MB each time the button is pressed
    NSMutableData* largeData = [[NSMutableData alloc] init];
    [largeData setLength:1024 * 1024 * 300]; 
    static int count = 2;
    NSString* key = [NSString stringWithFormat:@"test_data_%d", count++];
    [_testCache setObject:largeData forKey:key];

    NSMutableData* dataRecoveredFromCache = [_testCache objectForKey:@"original_Data"];

    if (dataRecoveredFromCache) {
        NSLog(@"Original data is ok");
    } else {
        NSLog(@"Original data is missing (purged from cache)");
    }
}

所以我在模拟器中运行了该应用程序,并点击了几次按钮,但没有任何项目被驱逐...该应用程序最终崩溃了:

2012-07-17 14:19:36.877 NSCacheTest[15302:f803] Data is ok
2012-07-17 14:19:37.365 NSCacheTest[15302:f803] Data is ok
2012-07-17 14:19:37.861 NSCacheTest[15302:f803] Data is ok
2012-07-17 14:19:38.341 NSCacheTest[15302:f803] Data is ok
2012-07-17 14:19:38.821 NSCacheTest[15302:f803] Data is ok
NSCacheTest(15302,0xac0942c0) malloc: *** mmap(size=393216000) failed (error code=12)
*** error: can't allocate region

【问题讨论】:

  • 我添加了一个UIApplicationDidReceiveMemoryWarningNotification 观察者,并在其中清空缓存。

标签: iphone objective-c ios nscache


【解决方案1】:

来自文档(我的重点):NSCache 类包含各种自动删除策略,以确保它不会使用过多的系统内存。如果其他应用程序需要内存,系统会自动执行这些策略。调用时,这些策略会从缓存中删除一些项目,从而最大限度地减少其内存占用。

Apple 并未声明内存警告时将释放内存 - 根据我的经验,当应用程序进入后台或添加更多大型元素时,缓存最常被清除。

【讨论】:

  • 谢谢你。我发现简单地将应用程序放到模拟器的后台会导致Data is missing。但是,即使通过不断分配大块,我也无法让它丢弃内存。有什么想法吗?
  • 事实上,无论数据多么小,锁定或将应用程序置于后台都会导致项目被驱逐。然而,不断分配 50 mb 块并将它们存储在缓存中并不能释放它们……最终应用程序崩溃了!
  • 有趣,我没有观察到这样的行为——你是在模拟器上还是在设备上测试?
  • 我确实做到了,两者都......我已经用我使用的源代码调整了我的问题。 (虽然我在设备上使用了较小的分配)
  • @Robert 您可以将“成本”与每个缓存对象相关联,并在 NSCache 对象上设置成本限制。假设您将 cost 设置为 data.length,因此您可以限制分配的内存。
【解决方案2】:

这是引用的文档...

NSCache 类包含各种自动删除策略,其中 确保它不会使用过多的系统内存。这 如果内存需要,系统会自动执行这些策略 其他应用。调用时,这些政策会删除 一些 项 从缓存中提取,最大限度地减少其内存占用。

...如您所见,它声明它删除了一些项,而不是所有项。这取决于NSCache 内部政策、可用内存、设备状态等。您不必担心这些政策。

您可以使用countLimittotalCostLimit 属性来控制它们,还可以使用成本添加对象,请查看setObject:forKey:cost:

你也可以自己驱逐对象。将NSDiscardableContent 协议实现添加到您的对象和setEvictsObjectsWithDiscardedContent: 到YES

【讨论】:

  • 我想说的是,即使你在模拟器中发出内存警告,NSCache 仍然可以认为(=知道)有很多空闲内存并且什么都不做。
【解决方案3】:

我也在使用那个类。请注意,文档指出 NSCache 与操作系统相关联,并且可能可以访问操作系统深处的内存信息。内存警告就是这样 - 它只是向 appDelegate/viewControllers 发送内存警告。

如果你真的想测试你的代码,你可能需要一个测试模式,你开始分配大量内存(可以说造成巨大的泄漏)。您可能需要在每个主运行循环期间将其分块打包,因此操作系统有机会看到他的内存下降(我有一个应用程序会占用大量内存,并且它在 3GS 上执行速度如此之快以至于被杀死从未收到内存警告。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2014-08-02
    • 2021-07-23
    • 1970-01-01
    • 2012-03-17
    • 1970-01-01
    • 2018-03-23
    • 1970-01-01
    • 2014-06-21
    相关资源
    最近更新 更多