如果一个应用中需要从云端加载较多图片,甚至会频繁刷新页面,那么这个应用就有必要将图片缓存起来,不用每次刷新都重新下载图片。
我这里写了一个简单的图片缓存工具,还比较实用。
头文件 ImageCache.h
#import <Foundation/Foundation.h>
@interface ImageCache : NSObject {
//缓存的服务器端图片
NSMutableDictionary *cachedRemoteImgs;
}
/**
加载图片
*/
- (BOOL)loadRemoteImg:(NSString *)imgUrl;
/**
从缓存中获取图片
*/
- (UIImage *)remoteImgForUrl:(NSString *)imgUrl;
@end
实现文件 ImageCache.m
#import "ImageCache.h"
@implementation ImageCache
- (id)init {
if (![super init]) {
return nil;
}
cachedRemoteImgs = [[NSMutableDictionary alloc] init];
return self;
}
- (BOOL)loadRemoteImg:(NSString *)imgUrl {
// NSLog(@"load img %@", imgUrl);
if (!cachedRemoteImgs) {
cachedRemoteImgs = [[NSMutableDictionary alloc] init];
}
if ([cachedRemoteImgs objectForKey:imgUrl]) {
// NSLog(@"this image is cached : %@", imgUrl);
return YES;
}
NSURL *url = [NSURL URLWithString:imgUrl];
NSData *imgData = [NSData dataWithContentsOfURL:url];
// NSLog(@"imgData size %d, url %@", imgData.length, imgUrl);
UIImage *img = [UIImage imageWithData:imgData];
if (img) {
[cachedRemoteImgs setObject:img forKey:imgUrl];
}
return YES;
}
- (UIImage *)remoteImgForUrl:(NSString *)imgUrl {
return (UIImage *)[cachedRemoteImgs objectForKey:imgUrl];
}
@end
在使用时最好在后台线程调用- (BOOL)loadRemoteImg:(NSString *)imgUrl 方法,因为加载图片会比较耗时,以免阻塞主线程。这里为什么没有直接在该方法中启后台线程加载图片呢?因为调用者一般需要在图片加载成功后做相应的操作。
比如:
if ([imageCache loadRemoteImg:imgUrl]) {
[self.leftTableView performSelectorOnMainThread:@selector(reloadData) withObject:nil waitUntilDone:NO];
}
说明:这是在某个后台线程加载图片,加载成功后通知表格在主线程中刷新数据。
很明显,在表格数据源的委托方法通过 [imageCache remoteImgForUrl:imgUrl] 去获取缓存好图片。