gpt4 book ai didi

ios - NSCache 持有指向用 imageWithData : and does not remove from memory on unload 实例化的 UIImage 的强指针

转载 作者:行者123 更新时间:2023-12-01 16:42:27 26 4
gpt4 key购买 nike

我有一个带有属性的 View Controller galleryCache当使用 GCD 和 imageWithData: 下载图像时图像已使用 key 成功添加到缓存中。但是,当 View Controller 被解除时,它会保留指向那些下载图像的强指针,从而导致它们不会从内存中删除。即使我使用 removeAllObjects viewDidDisappear: 中缓存的方法内存没有清理干净。

有谁知道这可能是为什么?

这是下载图像的方法的代码。

- (void)imageForFootageSize:(FootageSize)footageSize withCompletionHandler:(void (^)(UIImage *image))completionBlock
{
if (completionBlock) {
__block UIImage *image;

// Try getting local image from disk.
//
__block NSURL *imageURL = [self localURLForFootageSize:footageSize];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
image = [UIImage imageWithData:[NSData dataWithContentsOfURL:imageURL]];

dispatch_async(dispatch_get_main_queue(), ^{
if (image) {
completionBlock(image);
} else {
//
// Otherwise try getting remote image.
//
imageURL = [self remoteURLForFootageSize:footageSize];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSData *imageData = [NSData dataWithContentsOfURL:imageURL];

dispatch_async(dispatch_get_main_queue(), ^{
image = [UIImage imageWithData:imageData];

if (image) {
//
// Save remote image to disk
//
NSURL *photoDirectoryURL = [Footage localURLForDirectory];

// Create the folder(s) where the photos are stored.
//
[[NSFileManager defaultManager] createDirectoryAtPath:[photoDirectoryURL path] withIntermediateDirectories:YES attributes:nil error:nil];

// Save photo
//
NSString *localPath = [[self localURLForFootageSize:footageSize] path];
[imageData writeToFile:localPath atomically:YES];
}

completionBlock(image);
});
});
}
});
});
}
}

使用上述类方法获取和处理 UIImage 的方法在完成处理程序中。

内部方法 UICollectionViewCell子类。
- (void)setPhoto:(Photo *)photo withImage:(UIImage *)image
{
[self setBackgroundColor:[UIColor blackColor]];
[self.imageView setBackgroundColor:[UIColor clearColor]];

if (photo && !image) {
[photo imageForFootageSize:[Footage footageSizeThatBestFitsRect:self.bounds]
withCompletionHandler:^(UIImage *image) {
if ([self.delegate respondsToSelector:@selector(galleryPhotoCollectionViewCell:didLoadImage:)]) {
[self.delegate galleryPhotoCollectionViewCell:self didLoadImage:image];
}

image = nil;
}];
}

[self.imageView setImage:image];

BOOL isPhotoAvailable = (BOOL)(image);

[self.imageView setHidden:!isPhotoAvailable];
[self.activityIndicatorView setHidden:isPhotoAvailable];
}
UICollectionView 中的方法数据源委托(delegate)
- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
DIGalleryPhotoCollectionViewCell *photoCell = [collectionView dequeueReusableCellWithReuseIdentifier:photoCellIdentifier forIndexPath:indexPath];

[photoCell setDelegate:self];

Footage *footage = [self footageForIndexPath:indexPath];
Photo *photo = ([footage isKindOfClass:[Photo class]]) ? (Photo *)footage : nil;

if (photo) {
//
// Photo
//
[photoCell setPhoto:photo withImage:[self.galleryCache objectForKey:photo.footageID]];
}

return photoCell;
}

以下是其他相关方法:
- (void)galleryPhotoCollectionViewCell:(DIGalleryPhotoCollectionViewCell *)cell didLoadImage:(UIImage *)image
{
NSIndexPath *indexPath = [self.galleryCollectionView indexPathForCell:cell];

Footage *footage = [self footageForIndexPath:indexPath];

if ([footage isKindOfClass:[Footage class]]) {
Photo *photo = (Photo *)footage;

UIImage *cachedImage = [self.galleryCache objectForKey:photo.footageID];

if (!cachedImage) {
cachedImage = image;

[self.galleryCache setObject:image forKey:photo.footageID];
}

[cell setPhoto:photo withImage:image];
}
}

还有我的 NSCache 的 getter 方法房产 galleryCache
- (NSCache *)galleryCache
{
if (!_galleryCache) {
_galleryCache = [[NSCache alloc] init];
}

return _galleryCache;
}

编辑

这是 Instruments 的快照,显示了一个 NSCache 的所有者( View Controller )被解除后的保留计数历史记录。

最佳答案

我在这里没有看到任何明显的东西,尽管我建议在清除缓存的地方设置一个断点,并确保它确实像你想象的那样发生。

如果还是找不到,可以在 Instruments 中运行 Allocations 工具并打开“记录引用计数”(参见本答案的后半部分,iOS app with ARC, find who is owner of an object),您就可以准确找出您挥之不去的强引用在哪里,在这一点上,您可以解决补救问题。

另一个明显的解决方案是消除所有这些代码并使用经过验证的图像缓存工具,例如 SDWebImage它为您做了很多内存和持久存储缓存。这是一个相当不错的实现。

关于ios - NSCache 持有指向用 imageWithData : and does not remove from memory on unload 实例化的 UIImage 的强指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23045412/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com