设计一个移动应用的本地缓存机制
在手机应用程序开发中,为了降低与服务端的交互次数,加快用户的响应速度,一般都会在iOS设备中加一个缓存的机制,前面一篇文章介绍了iOS设备的内存缓存。这篇文章将设计一个本地缓存的机制。
功能需求
这个缓存机制满足以下这些功能。
1、能够将数据缓存到本地磁盘。
2、能够推断一个资源是否已经被缓存。假设已经被缓存。在请求同样的资源。先到本地磁盘搜索。
3、能够推断文件缓存什么时候过期。这里为了简单起见这里,我们在请求url资源的时候。给每次请求的文件设定一个过期的时间。
4、能够实现:假设文件已经被缓存,并且没有过期。这将本地的数据返回,否则又一次请求url。
5、能够实现:假设文件下载不成功或者下载没有完毕,下次打开程序的时候,移除这些没有成功或者没有下载完毕的文件。
6、能够实现:同一时候请求或者下载多个资源。
设计实现:
1、设计一个CacheItem类,用来请求一个web连接,它的一个实例表示一个缓存项。
这个CacheItem类,须要一个url创建一个NSURLConnection,去请求web资源。
使用CacheItem类主要用来请求web资源。
- /* ---------缓存项-------------- */
- @interface CacheItem : NSObject {
- @public
- id<CacheItemDelegate> delegate;
- //web地址
- NSString *remoteURL;
- @private
- //是否正在下载
- BOOL isDownloading;
- //NSMutableData对象
- NSMutableData *connectionData;
- //NSURLConnection对象
- NSURLConnection *connection;
- }
- /* -------------------------- */
- @property (nonatomic, retain) id<CacheItemDelegate> delegate;
- @property (nonatomic, retain) NSString *remoteURL;
- @property (nonatomic, assign) BOOL isDownloading;
- @property (nonatomic, retain) NSMutableData *connectionData;
- @property (nonatomic, retain) NSURLConnection *connection;
- /* ----------開始下载方法----------- */
- - (BOOL) startDownloadingURL:(NSString *)paramRemoteURL;
- @end
2、在NSURLConnection開始请求之前,调用CachedDownloadManager类,来搜索和管理本地的缓存文件。将缓存文件的情况保存到一个字典类中。
这个字典设计例如以下:
- {
- "http://www.cnn.com" = {
- DownloadEndDate = "2011-08-02 07:51:57 +0100";
- DownloadStartDate = "2011-08-02 07:51:55 +0100";
- ExpiresInSeconds = 20;
- ExpiryDate = "2011-08-02 07:52:17 +0100";
- LocalURL = "/var/mobile/Applications/ApplicationID/Documents/
- httpwww.cnn.com.cache";
- };
- "http://www.baidu.com" = {
- DownloadEndDate = "2011-08-02 07:51:49 +0100";
- DownloadStartDate = "2011-08-02 07:51:44 +0100";
- ExpiresInSeconds = 20;
- ExpiryDate = "2011-08-02 07:52:09 +0100";
- LocalURL = "/var/mobile/Applications/ApplicationID/Documents/
- httpwww.oreilly.com.cache";
- };
- }
上面这个字典里面嵌套了字典。里面那层字典表示一个缓存项的缓存信息:下载结束时间、下载開始时间、缓存有效时间、缓存过期时间、缓存到本地的路径。
以下看下CachedDownloadManager类。
用它来实现和封装我们的缓存策略。
- /* -----------CachedDownloadManager-------------- */
- @interface CachedDownloadManager : NSObject
- <CacheItemDelegate> {
- @public
- id<CachedDownloadManagerDelegate> delegate;
- @private
- //记录缓存数据的字典
- NSMutableDictionary *cacheDictionary;
- //缓存的路径
- NSString *cacheDictionaryPath;
- }
- @property (nonatomic, assign)
- id<CachedDownloadManagerDelegate> delegate;
- @property (nonatomic, copy)
- NSMutableDictionary *cacheDictionary;
- @property (nonatomic, retain)
- NSString *cacheDictionaryPath;
- /* 保持缓存字典 */
- - (BOOL) saveCacheDictionary;
- /* 公有方法:下载 */
- - (BOOL) download:(NSString *)paramURLAsString
- urlMustExpireInSeconds:(NSTimeInterval)paramURLMustExpireInSeconds
- updateExpiryDateIfInCache:(BOOL)paramUpdateExpiryDateIfInCache;
- /* -------------------------- */
- @end
从上面代码能够看出,这个管理缓存的类中。有一个缓存字典:cacheDictionary,用来表示全部资源的缓存情况;cacheDictionaryPath用来表示缓存的路径。saveCacheDictionary用来将缓存字典归档到本地文件里。
download:urlMustExpireInSeconds:updateExpiryDateIfInCache是一个公共接口,通过传递url、缓存过期时间、是否更新缓存过期时间三个參数来方便的使用,实现我们的缓存策略。
3、假设这个文件已经被下载,并且没有过期。则从本地获取文件的数据。假设文件已经过期。则又一次下载。
我们通过download:urlMustExpireInSeconds:updateExpiryDateIfInCache方法来实现,主要看这种方法的代码:
- /* ---------下载-------------- */
- - (BOOL) download:(NSString *)paramURLAsString
- urlMustExpireInSeconds:(NSTimeInterval)paramURLMustExpireInSeconds
- updateExpiryDateIfInCache:(BOOL)paramUpdateExpiryDateIfInCache{
- BOOL result = NO;
- if (self.cacheDictionary == nil ||
- [paramURLAsString length] == 0){
- return(NO);
- }
- paramURLAsString = [paramURLAsString lowercaseString];
- //依据url。从字典中获取缓存项的相关数据
- NSMutableDictionary *itemDictionary =
- [self.cacheDictionary objectForKey:paramURLAsString];
- /* 使用以下这些变量帮助我们理解缓存逻辑 */
- //文件是否已经被缓存
- BOOL fileHasBeenCached = NO;
- //缓存是否过期
- BOOL cachedFileHasExpired = NO;
- //缓存文件是否存在
- BOOL cachedFileExists = NO;
- //缓存文件是否能被载入
- BOOL cachedFileDataCanBeLoaded = NO;
- //缓存文件数据
- NSData *cachedFileData = nil;
- //缓存文件是否全然下载
- BOOL cachedFileIsFullyDownloaded = NO;
- //缓存文件是否已经下载
- BOOL cachedFileIsBeingDownloaded = NO;
- //过期时间
- NSDate *expiryDate = nil;
- //下载结束时间
- NSDate *downloadEndDate = nil;
- //下载開始时间
- NSDate *downloadStartDate = nil;
- //本地缓存路径
- NSString *localURL = nil;
- //有效时间
- NSNumber *expiresInSeconds = nil;
- NSDate *now = [NSDate date];
- if (itemDictionary != nil){
- fileHasBeenCached = YES;
- }
- //假设文件已经被缓存。则从缓存项相关数据中获取相关的值
- if (fileHasBeenCached == YES){
- expiryDate = [itemDictionary
- objectForKey:CachedKeyExpiryDate];
- downloadEndDate = [itemDictionary
- objectForKey:CachedKeyDownloadEndDate];
- downloadStartDate = [itemDictionary
- objectForKey:CachedKeyDownloadStartDate];
- localURL = [itemDictionary
- objectForKey:CachedKeyLocalURL];
- expiresInSeconds = [itemDictionary
- objectForKey:CachedKeyExpiresInSeconds];
- //假设下载開始和结束时间不为空,表示文件所有被下载
- if (downloadEndDate != nil &&
- downloadStartDate != nil){
- cachedFileIsFullyDownloaded = YES;
- }
- /* 假设expiresInSeconds不为空。downloadEndDate为空。表示文件已经正在下载 */
- if (expiresInSeconds != nil &&
- downloadEndDate == nil){
- cachedFileIsBeingDownloaded = YES;
- }
- /* 推断缓存是否过期 */
- if (expiryDate != nil &&
- [now timeIntervalSinceDate:expiryDate] > 0.0){
- cachedFileHasExpired = YES;
- }
- if (cachedFileHasExpired == NO){
- /* 假设缓存文件没有过期,载入缓存文件,而且更新过期时间 */
- NSFileManager *fileManager = [[NSFileManager alloc] init];
- if ([fileManager fileExistsAtPath:localURL] == YES){
- cachedFileExists = YES;
- cachedFileData = [NSData dataWithContentsOfFile:localURL];
- if (cachedFileData != nil){
- cachedFileDataCanBeLoaded = YES;
- } /* if (cachedFileData != nil){ */
- } /* if ([fileManager fileExistsAtPath:localURL] == YES){ */
- [fileManager release];
- /* 更新缓存时间 */
- if (paramUpdateExpiryDateIfInCache == YES){
- NSDate *newExpiryDate =
- [NSDate dateWithTimeIntervalSinceNow:
- paramURLMustExpireInSeconds];
- NSLog(@"Updating the expiry date from %@ to %@.",
- expiryDate,
- newExpiryDate);
- [itemDictionary setObject:newExpiryDate
- forKey:CachedKeyExpiryDate];
- NSNumber *expires =
- [NSNumber numberWithFloat:paramURLMustExpireInSeconds];
- [itemDictionary setObject:expires
- forKey:CachedKeyExpiresInSeconds];
- }
- } /* if (cachedFileHasExpired == NO){ */
- }
- if (cachedFileIsBeingDownloaded == YES){
- NSLog(@"这个文件已经正在下载...");
- return(YES);
- }
- if (fileHasBeenCached == YES){
- if (cachedFileHasExpired == NO &&
- cachedFileExists == YES &&
- cachedFileDataCanBeLoaded == YES &&
- [cachedFileData length] > 0 &&
- cachedFileIsFullyDownloaded == YES){
- /* 假设文件有缓存并且没有过期 */
- NSLog(@"文件有缓存并且没有过期.");
- [self.delegate
- cachedDownloadManagerSucceeded:self
- remoteURL:[NSURL URLWithString:paramURLAsString]
- localURL:[NSURL URLWithString:localURL]
- aboutToBeReleasedData:cachedFileData
- isCachedData:YES];
- return(YES);
- } else {
- /* 假设文件没有被缓存。获取缓存失败 */
- NSLog(@"文件没有缓存.");
- [self.cacheDictionary removeObjectForKey:paramURLAsString];
- [self saveCacheDictionary];
- } /* if (cachedFileHasExpired == NO && */
- } /* if (fileHasBeenCached == YES){ */
- /* 去下载文件 */
- NSNumber *expires =
- [NSNumber numberWithFloat:paramURLMustExpireInSeconds];
- NSMutableDictionary *newDictionary =
- [[[NSMutableDictionary alloc] init] autorelease];
- [newDictionary setObject:expires
- forKey:CachedKeyExpiresInSeconds];
- localURL = [paramURLAsString
- stringByAddingPercentEscapesUsingEncoding:
- NSUTF8StringEncoding];
- localURL = [localURL stringByReplacingOccurrencesOfString:@"://"
- withString:@""];
- localURL = [localURL stringByReplacingOccurrencesOfString:@"/"
- withString:@"{1}quot;];
- localURL = [localURL stringByAppendingPathExtension:@"cache"];
- NSString *documentsDirectory =
- [self documentsDirectoryWithTrailingSlash:NO];
- localURL = [documentsDirectory
- stringByAppendingPathComponent:localURL];
- [newDictionary setObject:localURL
- forKey:CachedKeyLocalURL];
- [newDictionary setObject:now
- forKey:CachedKeyDownloadStartDate];
- [self.cacheDictionary setObject:newDictionary
- forKey:paramURLAsString];
- [self saveCacheDictionary];
- CacheItem *item = [[[CacheItem alloc] init] autorelease];
- [item setDelegate:self];
- [item startDownloadingURL:paramURLAsString];
- return(result);
- }
4、以下我们设计缓存项下载成功和失败的两个托付方法:
- @protocol CacheItemDelegate <NSObject>
- //下载成功运行该方法
- - (void) cacheItemDelegateSucceeded
- :(CacheItem *)paramSender
- withRemoteURL:(NSURL *)paramRemoteURL
- withAboutToBeReleasedData:(NSData *)paramAboutToBeReleasedData;
- //下载失败运行该方法
- - (void) cacheItemDelegateFailed
- :(CacheItem *)paramSender
- remoteURL:(NSURL *)paramRemoteURL
- withError:(NSError *)paramError;
- @end
当我们下载成功的时候,改动缓存字典中的下载时间,表示已经下载完毕。并且须要将请求的资源数据缓存到本地:
- //缓存项的托付方法
- - (void) cacheItemDelegateSucceeded:(CacheItem *)paramSender
- withRemoteURL:(NSURL *)paramRemoteURL
- withAboutToBeReleasedData:(NSData *)paramAboutToBeReleasedData{
- //从缓存字典中获取该缓存项的相关数据
- NSMutableDictionary *dictionary =
- [self.cacheDictionary objectForKey:[paramRemoteURL absoluteString]];
- //取当前时间
- NSDate *now = [NSDate date];
- //获取有效时间
- NSNumber *expiresInSeconds = [dictionary
- objectForKey:CachedKeyExpiresInSeconds];
- //转换成NSTimeInterval
- NSTimeInterval expirySeconds = [expiresInSeconds floatValue];
- //改动字典中缓存项的下载结束时间
- [dictionary setObject:[NSDate date]
- forKey:CachedKeyDownloadEndDate];
- //改动字典中缓存项的缓存过期时间
- [dictionary setObject:[now dateByAddingTimeInterval:expirySeconds]
- forKey:CachedKeyExpiryDate];
- //保存缓存字典
- [self saveCacheDictionary];
- NSString *localURL = [dictionary objectForKey:CachedKeyLocalURL];
- /* 将下载的数据保持到磁盘 */
- if ([paramAboutToBeReleasedData writeToFile:localURL
- atomically:YES] == YES){
- NSLog(@"缓存文件到磁盘成功.");
- } else{
- NSLog(@"缓存文件到磁盘失败.");
- }
- //运行缓存管理的托付方法
- [self.delegate
- cachedDownloadManagerSucceeded:self
- remoteURL:paramRemoteURL
- localURL:[NSURL URLWithString:localURL]
- aboutToBeReleasedData:paramAboutToBeReleasedData
- isCachedData:NO];
- }
- //缓存项失败失败的托付方法
- - (void) cacheItemDelegateFailed:(CacheItem *)paramSender
- remoteURL:(NSURL *)paramRemoteURL
- withError:(NSError *)paramError{
- /* 从缓存字典中移除缓存项,并发送一个托付 */
- if (self.delegate != nil){
- NSMutableDictionary *dictionary =
- [self.cacheDictionary
- objectForKey:[paramRemoteURL absoluteString]];
- NSString *localURL = [dictionary
- objectForKey:CachedKeyLocalURL];
- [self.delegate
- cachedDownloadManagerFailed:self
- remoteURL:paramRemoteURL
- localURL:[NSURL URLWithString:localURL]
- withError:paramError];
- }
- [self.cacheDictionary
- removeObjectForKey:[paramRemoteURL absoluteString]];
- }
- //初始化缓存字典
- NSString *documentsDirectory =
- [self documentsDirectoryWithTrailingSlash:YES];
- //生产缓存字典的路径
- cacheDictionaryPath =
- [[documentsDirectory
- stringByAppendingString:@"CachedDownloads.dic"] retain];
- //创建一个NSFileManager实例
- NSFileManager *fileManager = [[NSFileManager alloc] init];
- //推断是否存在缓存字典的数据
- if ([fileManager
- fileExistsAtPath:self.cacheDictionaryPath] == YES){
- NSLog(self.cacheDictionaryPath);
- //载入缓存字典中的数据
- NSMutableDictionary *dictionary =
- [[NSMutableDictionary alloc]
- initWithContentsOfFile:self.cacheDictionaryPath];
- cacheDictionary = [dictionary mutableCopy];
- [dictionary release];
- //移除没有下载完毕的缓存数据
- [self removeCorruptedCachedItems];
- } else {
- //创建一个新的缓存字典
- NSMutableDictionary *dictionary =
- [[NSMutableDictionary alloc] init];
- cacheDictionary = [dictionary mutableCopy];
- [dictionary release];
- }
这样就基本上完毕了我们须要的功能。以下看看我们怎样使用我们设计的缓存功能。
样例场景:
我们用一个UIWebView来显示stackoverflow这个站点,我们在这个站点的内容缓存到本地20秒,假设在20秒内用户去请求该站点,则从本地文件里获取内容,否则过了20秒。则又一次获取数据,并缓存到本地。
在界面上拖放一个button和一个webview控件,例如以下图。
这样我们能够非常方便使用前面定义好的类。
我们在viewDidLoad 中实例化一个CachedDownloadManager,并设置它的托付为self。
当下载完毕的时候。运行CachedDownloadManager的下载成功的托付方法。
- (void)viewDidLoad {[super viewDidLoad]; [self setTitle:@"本地缓存測试"];CachedDownloadManager *newManager = [[CachedDownloadManager alloc] init];self.downloadManager = newManager;[newManager release];[self.downloadManager setDelegate:self];}
在button的点击事件中增加以下代码,请求stackoverflow :
static NSString *url = @"http://stackoverflow.com";[self.downloadManager download:urlurlMustExpireInSeconds:20.0fupdateExpiryDateIfInCache:YES];
上面的代码表示将这个stackoverflow的缓存事件设置为20s。而且假设在20s内有同样的请求,则从本地获取stackoverflow的内容数据。updateExpiryDateIfInCache设置为yes表示:在此请求的时候。缓存时间又更新为20s。类似我们的session。假设设置成no。则第一次请求20s之后。该缓存就过期。
请求完毕之后会运行CachedDownloadManager的托付方法。我们将数据展示在uiwebview中,代码例如以下:
- (void) cachedDownloadManagerSucceeded:(CachedDownloadManager *)paramSenderremoteURL:(NSURL *)paramRemoteURLlocalURL:(NSURL *)paramLocalURLaboutToBeReleasedData:(NSData *)paramAboutToBeReleasedDataisCachedData:(BOOL)paramIsCachedData{ [webview loadData:paramAboutToBeReleasedData MIMEType:@"text/html" textEncodingName:@"UTF-8" baseURL:[NSURL URLWithString:@"http://stackoverflow.com"]]; }
这样我们就实现了20s的缓存。
效果:
第一次点击測试button:
20s内点击button,程序就从本地获取数据。比較高速的就显示出该网页了。
总结:
本文通过代码和实例设计了一个iPhone应用程序本地缓存的方案。
当然这个方法不是最好的,假设你有更好的思路。欢迎告诉我。
设计一个移动应用的本地缓存机制相关推荐
- 如何设计一个牛逼的本地缓存
点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 来源:ksfzhaohui juejin.im/post/5dd9 ...
- 如何设计一个牛逼的本地缓存!
来源:ksfzhaohui | http://dwz.win/Ws4 最近在看Mybatis的源码,刚好看到缓存这一块,Mybatis提供了一级缓存和二级缓存:一级缓存相对来说比较简单,功能比较齐全的 ...
- 我设计了一个牛逼的本地缓存!
点击上方 好好学java ,选择 星标 公众号 重磅资讯.干货,第一时间送达 今日推荐:硬刚一周,3W字总结,一年的经验告诉你如何准备校招! 个人原创100W+访问量博客:点击前往,查看更多 作者:k ...
- 多任务学习,如何设计一个更好的参数共享机制?| AAAI 2020
2019-12-26 05:44:43 作者 | 孙天祥 编辑 | 刘萍 原文标题:稀疏共享:当多任务学习遇见彩票假设 本文介绍了复旦大学邱锡鹏团队在AAAI 2020 上录用的一篇关于多任务学习的工 ...
- GUAVA本地缓存01_概述、优缺点、创建方式、回收机制、监听器、统计、异步锁定
文章目录 ①. 本地缓存 - 背景 ②. 本地缓存 - 优缺点 ③. Guava Cache介绍 ④. Guava - 三种创建方式 ⑤. Guava - 如何回收缓存 ⑥. Guava - 移除监听 ...
- 本地缓存-loadingCache
本地缓存 缓存大概是一个不能再熟悉的话题,今天的主要内容是分享一下我们公司使用比较多的本地缓存 loadingcache,同时也是自己使用过程中的一些探索的分享,其背后的架构其实就是Guava cac ...
- 实现 Java 本地缓存
缓存,我相信大家对它一定不陌生,在项目中,缓存肯定是必不可少的.市面上有非常多的缓存工具,比如 Redis.Guava Cache 或者 EHcache.对于这些工具,我想大家肯定都非常熟悉,所以今天 ...
- java几点钟_实现 Java 本地缓存,该从这几点开始
缓存,我相信大家对它一定不陌生,在项目中,缓存肯定是必不可少的.市面上有非常多的缓存工具,比如 Redis.Guava Cache 或者 EHcache. 对于这些工具,我想大家肯定都非常熟悉,所以今 ...
- Guava 本地缓存使用教程
文章目录 准备工作 创建缓存 使用缓存 Cache 读取缓存 LoadingCache 读取缓存 修改缓存 Cache 修改缓存 LoadingCache 修改缓存 其他方法 在前面的文章 Sprin ...
最新文章
- Imc连环画《红楼梦》
- 2018.11.09 codeforces487E. Tourists(tarjan+树链剖分)
- Linux下svn新建用户,Linux下建立svn工程
- codematic2连接mysql失败_codematic2.rar
- 高级数据结构与算法 | 回溯算法(Back Tracking Method)
- android编译的错误日志,android编译遇到错误
- Android View框架总结(八)ViewGroup事件分发机制
- 串口通讯WaitCommEvent 、GetLastError、ClearCommError、...
- 【环境搭建】zip 分卷压缩
- python删除第一行_python学习之删除DataFrame某一行/列内容
- 读《Machine Learning in Action》的感想
- 李航·《统计学习方法》学习笔记
- 关于BufferedOutputStream的flush方法
- Java应用性能分析工具:async-profiler
- 差分进化算法_想用遗传算法?来看看这些已为你做好的开源优化框架
- gitbook 使用粘自csdn
- 罗马音平假名中文可复制_如何快速有效地学习日语五十音?
- python高级编程(6) - 对象引用,可变性和垃圾和回收
- 网件 R6400 TTL 救砖详细 教程
- PPt2007制作三维立体字效果教程
热门文章
- error while loading shared libraries: libgconf-2.so.4
- 获取windows所有端口
- 找到bug的根源,问五次为什么
- [leetcode]687. Longest Univalue Path
- lvs + keepalive的安装配置
- Web中常用字体介绍(转)
- 解决ng界面长表达式(ui-set)
- 在VS2010中F5调试Silverlight程序时,提示“无法启动调试,找不到Microsoft Internet Explorer”...
- SYSTEM32 下的几乎所有文件的简单说明
- 最新!全球学术排名出炉:22所中国大学位居世界100强