为什么要加入离线缓存
对同一个网络接口请求多次,返回的数据可能都是一样的,比如服务器上的某张图片,无论下载多少次,返回的数据都是一样的。
其次在网络环境很差的条件下,用户仍然可以浏览一些列表数据。
比如慕课网和大众点评接口都是标记的,接口返回数据是否需要更新,客户端再去请求数据等,减少用户流量,使客户端拥有更好的体验。
数据展示型的页面做离线缓存可以有更好的用户体验,用户在离线环境下仍然可以获取一些数据。
使用何种方式存储数据
这里的数据缓存首选是SQLite,轻量级,对数据的存储读取相对于其他几种方式有优势。
SQLite是一款轻型的嵌入式数据库,安卓和ios开发使用的都是SQLite数据库。
它占用资源非常的低,在嵌入式设备中,可能只需要几百K的内存就够了。
它的处理速度比Mysql、PostgreSQL这两款著名的数据库都还快。
离线缓存的思路:
需要注意的点:
在对自定义对象进行归档的时候,使用MJExtension归档。不需要写任何归档代码。一句话足够了。
MJCodingImplementation
#import <Foundation/Foundation.h>
#import "ProjectListModel.h"
@interface InterfaceDataBase : NSObject
+ (void)saveListData:(ListData *)listData;
+ (NSArray *)list;
+ (NSArray *)listWithRange:(NSRange)range;
+ (BOOL)isExistWithId:(NSString *)idStr;
@end
#import "InterfaceDataBase.h"
#import "FMDatabase.h"
#import "ProjectListModel.h"
@implementation InterfaceDataBase
static FMDatabase *_db;
+ (void)initialize {
NSString *path = [NSString stringWithFormat:@"%@/Library/Caches/Data.db",NSHomeDirectory()];
_db = [FMDatabase databaseWithPath:path];
[_db open];
[_db executeUpdate:@"CREATE TABLE IF NOT EXISTS t_project (id integer PRIMARY KEY, project blob NOT NULL, idStr text NOT NULL)"];
}
+ (void)saveListData:(ListData *)listData {
NSData *dictData = [NSKeyedArchiver archivedDataWithRootObject:listData];
[_db executeUpdateWithFormat:@"INSERT INTO t_project (project, idStr) VALUES (%@, %@)",dictData,listData.projcetid];
}
+ (NSArray *)list {
FMResultSet *set = [_db executeQuery:@"SELECT * FROM t_project"];
NSMutableArray *list = [NSMutableArray array];
while (set.next) {
NSData *dictData = [set objectForColumnName:@"project"];
ListData *listData = [NSKeyedUnarchiver unarchiveObjectWithData:dictData];
[list addObject:listData];
}
return list;
}
+ (NSArray *)listWithRange:(NSRange)range {
NSString *SQL = [NSString stringWithFormat:@"SELECT * FROM t_project LIMIT %lu, %lu",range.location, range.length];
FMResultSet *set = [_db executeQuery:SQL];
NSMutableArray *list = [NSMutableArray array];
while (set.next) {
NSData *dictData = [set objectForColumnName:@"project"];
ListData *listData = [NSKeyedUnarchiver unarchiveObjectWithData:dictData];
[list addObject:listData];
}
return list;
}
+ (BOOL)isExistWithId:(NSString *)idStr{
BOOL isExist = NO;
FMResultSet *resultSet= [_db executeQuery:@"SELECT * FROM t_project where idStr = ?",idStr];
while ([resultSet next]) {
if([resultSet stringForColumn:@"idStr"]) {
isExist = YES;
}else{
isExist = NO;
}
}
return isExist;
}
@end
具体使用
AFHTTPRequestOperationManager *manager=[AFHTTPRequestOperationManager manager];
manager.requestSerializer.timeoutInterval = 10.f;
NSRange range = NSMakeRange(0, 10*currentPage);
//如果数据库有数据则读取,不发送网络请求
if ([[InterfaceDataBase listWithRange:range] count]) {
[refreshArr addObjectsFromArray:[InterfaceDataBase listWithRange:range]];
[self.projectTableView reloadData];
NSLog(@"从数据库加载数据");
}else{
[ProjectRequest requestProjectListWithToken:currentUser.token
isauto:isAuto
isatten:@""
seach:self.searchText
statges:postStatgeStr
industrys:postIndustryStr
frounds:postFroundStr
pagenum:[NSString stringWithFormat:@"%ld",currentPage]
display_id:currentUser.displayId
username:currentUser.mobilePhone
cust_class:currentUser.identity
cust_id:currentUser.custId
manager:manager success:^(ProjectListModel * responseObject) {
[_baseView endRefreshing];
for (ListData *project in responseObject.data.data) {
//先判断数据是否存储过,如果没有,网络请求新数据存入数据库
if (![InterfaceDataBase isExistWithId:project.projcetid]) {
//存数据库
NSLog(@"存入数据库");
[InterfaceDataBase saveListData:project];
}
}
// [refreshArr addObjectsFromArray:responseObject.data.data];
// [self.projectTableView reloadData];
} failure:^(id responseObject) {
} failure:^(NSOperation *operation, NSError *error) {
}];
}