iOS数据库离线缓存思路和网络层封装

2024-06-11 14:38

本文主要是介绍iOS数据库离线缓存思路和网络层封装,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一直想总结一下关于iOS的离线数据缓存的方面的问题,然后最近也简单的对AFN进行了再次封装,所有想把这两个结合起来写一下。数据展示型的页面做离线缓存可以有更好的用户体验,用户在离线环境下仍然可以获取一些数据,这里的数据缓存首选肯定是SQLite,轻量级,对数据的存储读取相对于其他几种方式有优势,这里对AFN的封装没有涉及太多业务逻辑层面的需求,主要还是对一些方法再次封装方便使用,解除项目对第三方的耦合性,能够简单的快速的更换底层使用的网络请求代码。这篇主要写离线缓存思路,对AFN的封装只做简单的介绍。

关于XLNetworkApi

XLNetworkApi的一些功能和说明:

  • 使用XLNetworkRequest做一些GET、POST、PUT、DELETE请求,与业务逻辑对接部分直接以数组或者字典的形式返回。
  • 以及网络下载、上传文件,以block的形式返回实时的下载、上传进度,上传文件参数通过模型XLFileConfig去存取。
  • 通过继承于XLDataService来将一些数据处理,模型转化封装起来,于业务逻辑对接返回的是对应的模型,减少Controllor处理数据处理逻辑的压力。

  • 自定义一些回调的block

    /**
    请求成功block
    */
    typedef void (^requestSuccessBlock)(id responseObj);
    /**
    请求失败block
    */
    typedef void (^requestFailureBlock) (NSError *error);
    /**
    请求响应block
    */
    typedef void (^responseBlock)(id dataObj, NSError *error);
    /**
    监听进度响应block
    */
    typedef void (^progressBlock)(int64_t bytesWritten, int64_t totalBytesWritten, int64_t totalBytesExpectedToWrite);
  • XLNetworkRequest.m部分实现

    #import "XLNetworkRequest.h"
    #import "AFNetworking.h"
    @implementation XLNetworkRequest
    + (void)getRequest:(NSString *)url params:(NSDictionary *)params success:(requestSuccessBlock)successHandler failure:(requestFailureBlock)failureHandler {
    //网络不可用if (![self checkNetworkStatus]) {successHandler(nil);failureHandler(nil);return;}AFHTTPRequestOperationManager *manager = [self getRequstManager];[manager GET:url parameters:params success:^(AFHTTPRequestOperation * _Nonnull operation, id  _Nonnull responseObject) {successHandler(responseObject);} failure:^(AFHTTPRequestOperation * _Nullable operation, NSError * _Nonnull error) {XLLog(@"------请求失败-------%@",error);failureHandler(error);}];
    }
  • 下载部分代码

    //下载文件,监听下载进度
    + (void)downloadRequest:(NSString *)url successAndProgress:(progressBlock)progressHandler complete:(responseBlock)completionHandler {if (![self checkNetworkStatus]) {progressHandler(0, 0, 0);completionHandler(nil, nil);return;}NSURLSessionConfiguration *sessionConfiguration = [NSURLSessionConfiguration defaultSessionConfiguration];AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:sessionConfiguration];NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:url]];NSProgress *kProgress = nil;NSURLSessionDownloadTask *downloadTask = [manager downloadTaskWithRequest:request progress:&kProgress destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath, NSURLResponse * _Nonnull response) {NSURL *documentUrl = [[NSFileManager defaultManager] URLForDirectory :NSDocumentDirectory inDomain:NSUserDomainMask appropriateForURL:nil create:NO error:nil];return [documentUrl URLByAppendingPathComponent:[response suggestedFilename]];} completionHandler:^(NSURLResponse * _Nonnull response, NSURL * _Nonnull filePath, NSError * _Nonnull error){if (error) {XLLog(@"------下载失败-------%@",error);}completionHandler(response, error);}];[manager setDownloadTaskDidWriteDataBlock:^(NSURLSession * _Nonnull session, NSURLSessionDownloadTask * _Nonnull downloadTask, int64_t bytesWritten, int64_t totalBytesWritten, int64_t totalBytesExpectedToWrite) {progressHandler(bytesWritten, totalBytesWritten, totalBytesExpectedToWrite);}];[downloadTask resume];
    }
  • 上传部分代码

    //上传文件,监听上传进度
    + (void)updateRequest:(NSString *)url params:(NSDictionary *)params fileConfig:(XLFileConfig *)fileConfig successAndProgress:(progressBlock)progressHandler complete:(responseBlock)completionHandler {if (![self checkNetworkStatus]) {progressHandler(0, 0, 0);completionHandler(nil, nil);return;}NSMutableURLRequest *request = [[AFHTTPRequestSerializer serializer] multipartFormRequestWithMethod:@"POST" URLString:url parameters:params constructingBodyWithBlock:^(id<AFMultipartFormData>  _Nonnull formData) {[formData appendPartWithFileData:fileConfig.fileData name:fileConfig.name fileName:fileConfig.fileName mimeType:fileConfig.mimeType];} error:nil];//获取上传进度AFHTTPRequestOperation *operation = [[AFHTTPRequestOperation alloc] initWithRequest:request];[operation setUploadProgressBlock:^(NSUInteger bytesWritten, long long totalBytesWritten, long long totalBytesExpectedToWrite) {progressHandler(bytesWritten, totalBytesWritten, totalBytesExpectedToWrite);}];[operation setCompletionBlockWithSuccess:^(AFHTTPRequestOperation * _Nonnull operation, id  _Nonnull responseObject) {completionHandler(responseObject, nil);} failure:^(AFHTTPRequestOperation * _Nonnull operation, NSError * _Nonnull error) {completionHandler(nil, error);if (error) {XLLog(@"------上传失败-------%@",error);}}];[operation start];
    }
  • XLDataService.m部分实现

    + (void)getWithUrl:(NSString *)url param:(id)param modelClass:(Class)modelClass responseBlock:(responseBlock)responseDataBlock {[XLNetworkRequest getRequest:url params:param success:^(id responseObj) {//数组、字典转化为模型数组dataObj = [self modelTransformationWithResponseObj:responseObj modelClass:modelClass];responseDataBlock(dataObj, nil);} failure:^(NSError *error) {responseDataBlock(nil, error);}];
    }
  • (关键)下面这个方法提供给继承XLDataService的子类重写,将转化为模型的代码写在这里,相似业务的网络数据请求都可以用这个子类去请求数据,直接返回对应的模型数组。
    /**
    数组、字典转化为模型
    */
    + (id)modelTransformationWithResponseObj:(id)responseObj modelClass:(Class)modelClass {return nil;
    }
    关于离线数据缓存
    当用户进入程序的展示页面,有三个情况下可能涉及到数据库存取操作,简单画了个图来理解,思路比较简单,主要是一些存取的细节处理。
  • 进入展示页面


    进入页面.png
  • 下拉刷新最新数据


    下拉刷新.png
  • 上拉加载更多数据


    上拉加载更多.png
  • 需要注意的是,上拉加载更多的时候,每次从数据库返回一定数量的数据,而不是一次性将数据全部加载,否则会有内存问题,直到数据库中没有更多数据时再发生网络请求,再次将新数据存入数据库。这里存储数据的方式是将服务器返回每组数据的字典归档成二进制作为数据库字段直接存储,这样存储在模型属性比较多的情况下更有优势,避免每一个属性作为一个字段,另外增加了一个idStr字段用来判断数据的唯一性,避免重复存储。
    首先定义一个工具类XLDataBase来做数据库相关的操作,这里用的是第三方的FMDB。
#import "XLDataBase.h"
#import "FMDatabase.h"
#import "Item.h"
#import "MJExtension.h"@implementation XLDataBasestatic 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_item (id integer PRIMARY KEY, itemDict blob NOT NULL, idStr text NOT NULL)"];
}//存入数据库
+ (void)saveItemDict:(NSDictionary *)itemDict {//此处把字典归档成二进制数据直接存入数据库,避免添加过多的数据库字段NSData *dictData = [NSKeyedArchiver archivedDataWithRootObject:itemDict];[_db executeUpdateWithFormat:@"INSERT INTO t_item (itemDict, idStr) VALUES (%@, %@)",dictData, itemDict[@"id"]];
}//返回全部数据
+ (NSArray *)list {FMResultSet *set = [_db executeQuery:@"SELECT * FROM t_item"];NSMutableArray *list = [NSMutableArray array];while (set.next) {// 获得当前所指向的数据NSData *dictData = [set objectForColumnName:@"itemDict"];NSDictionary *dict = [NSKeyedUnarchiver unarchiveObjectWithData:dictData];[list addObject:[Item mj_objectWithKeyValues:dict]];}return list;
}//取出某个范围内的数据
+ (NSArray *)listWithRange:(NSRange)range {NSString *SQL = [NSString stringWithFormat:@"SELECT * FROM t_item LIMIT %lu, %lu",range.location, range.length];FMResultSet *set = [_db executeQuery:SQL];NSMutableArray *list = [NSMutableArray array];while (set.next) {NSData *dictData = [set objectForColumnName:@"itemDict"];NSDictionary *dict = [NSKeyedUnarchiver unarchiveObjectWithData:dictData];[list addObject:[Item mj_objectWithKeyValues:dict]];}return list;
}//通过一组数据的唯一标识判断数据是否存在
+ (BOOL)isExistWithId:(NSString *)idStr
{BOOL isExist = NO;FMResultSet *resultSet= [_db executeQuery:@"SELECT * FROM t_item where idStr = ?",idStr];while ([resultSet next]) {if([resultSet stringForColumn:@"idStr"]) {isExist = YES;}else{isExist = NO;}}return isExist;
}
@end
  • 一些继承于XLDataService的子类的数据库存储和模型转换的逻辑代码
#import "GetTableViewData.h"
#import "XLDataBase.h"@implementation GetTableViewData//重写父类方法
+ (id)modelTransformationWithResponseObj:(id)responseObj modelClass:(Class)modelClass {NSArray *lists = responseObj[@"data"][@"list"];NSMutableArray *array = [NSMutableArray array];for (NSDictionary *dict in lists) {[modelClass mj_setupReplacedKeyFromPropertyName:^NSDictionary *{return @{ @"ID" : @"id" };}];[array addObject:[modelClass mj_objectWithKeyValues:dict]];//通过idStr先判断数据是否存储过,如果没有,网络请求新数据存入数据库if (![XLDataBase isExistWithId:dict[@"id"]]) {//存数据库NSLog(@"存入数据库");[XLDataBase saveItemDict:dict];}}return array;
}
  • 下面是一些控制器的代码实现:
#import "ViewController.h"
#import "GetTableViewData.h"
#import "Item.h"
#import "XLDataBase.h"
#import "ItemCell.h"
#import "MJRefresh.h"
#define URL_TABLEVIEW @"https://api.108tian.com/mobile/v3/EventList?cityId=1&step=10&theme=0&page=%lu"@interface ViewController () <UITableViewDataSource, UITableViewDelegate>
{NSMutableArray *_dataArray;UITableView *_tableView;NSInteger _currentPage;//当前数据对应的page
}
@end@implementation ViewController
#pragma mark Life cycle
- (void)viewDidLoad {[super viewDidLoad];// Do any additional setup after loading the view, typically from a nib.[self createTableView];_dataArray = [NSMutableArray array];
}- (void)viewWillAppear:(BOOL)animated {[super viewWillAppear:animated];NSRange range = NSMakeRange(0, 10);//如果数据库有数据则读取,不发送网络请求if ([[XLDataBase listWithRange:range] count]) {[_dataArray addObjectsFromArray:[XLDataBase listWithRange:range]];NSLog(@"从数据库加载");}else{[self getTableViewDataWithPage:0];}
}#pragma mark UI
- (void)createTableView {_tableView = [[UITableView alloc] initWithFrame:self.view.bounds];_tableView.delegate = self;_tableView.dataSource = self;_tableView.rowHeight = 100.0;[self.view addSubview:_tableView];_tableView.mj_header = [MJRefreshNormalHeader headerWithRefreshingBlock:^{[self loadNewData];}];_tableView.mj_footer = [MJRefreshAutoNormalFooter footerWithRefreshingBlock:^{[self loadMoreData];}];
}#pragma mark GetDataSoure
- (void)getTableViewDataWithPage:(NSInteger)page {NSLog(@"发送网络请求!");NSString *url = [NSString stringWithFormat:URL_TABLEVIEW, page];[GetTableViewData getWithUrl:url param:nil modelClass:[Item class] responseBlock:^(id dataObj, NSError *error) {[_dataArray addObjectsFromArray:dataObj];[_tableView reloadData];[_tableView.mj_header endRefreshing];[_tableView.mj_footer endRefreshing];}];
}- (void)loadNewData {NSLog(@"下拉刷新");_currentPage = 0;[_dataArray removeAllObjects];[self getTableViewDataWithPage:_currentPage];
}- (void)loadMoreData {NSLog(@"上拉加载");_currentPage ++;NSRange range = NSMakeRange(_currentPage * 10, 10);if ([[XLDataBase listWithRange:range] count]) {[_dataArray addObjectsFromArray:[XLDataBase listWithRange:range]];[_tableView reloadData];[_tableView.mj_footer endRefreshing];NSLog(@"数据库加载%lu条更多数据",[[XLDataBase listWithRange:range] count]);}else{//数据库没更多数据时再网络请求[self getTableViewDataWithPage:_currentPage];}
}#pragma mark UITableViewDataSource
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {return _dataArray.count;
}- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {ItemCell *cell = [ItemCell itemCellWithTableView:tableView];cell.item = _dataArray[indexPath.row];return cell;
}
@end

最后附上代码的下载地址,重要的部分代码中都有相应的注释和文字打印,运行程序可以很直观的表现。

https://github.com/ShelinShelin/OffLineCache.git
有考虑不周的地方,希望大家能提出一些意见,很乐意与大家互相交流。

这篇关于iOS数据库离线缓存思路和网络层封装的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1051450

相关文章

MySQL数据库中ENUM的用法是什么详解

《MySQL数据库中ENUM的用法是什么详解》ENUM是一个字符串对象,用于指定一组预定义的值,并可在创建表时使用,下面:本文主要介绍MySQL数据库中ENUM的用法是什么的相关资料,文中通过代码... 目录mysql 中 ENUM 的用法一、ENUM 的定义与语法二、ENUM 的特点三、ENUM 的用法1

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

嵌入式数据库SQLite 3配置使用讲解

《嵌入式数据库SQLite3配置使用讲解》本文强调嵌入式项目中SQLite3数据库的重要性,因其零配置、轻量级、跨平台及事务处理特性,可保障数据溯源与责任明确,详细讲解安装配置、基础语法及SQLit... 目录0、惨痛教训1、SQLite3环境配置(1)、下载安装SQLite库(2)、解压下载的文件(3)、

Golang如何对cron进行二次封装实现指定时间执行定时任务

《Golang如何对cron进行二次封装实现指定时间执行定时任务》:本文主要介绍Golang如何对cron进行二次封装实现指定时间执行定时任务问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录背景cron库下载代码示例【1】结构体定义【2】定时任务开启【3】使用示例【4】控制台输出总结背景

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

MySQL追踪数据库表更新操作来源的全面指南

《MySQL追踪数据库表更新操作来源的全面指南》本文将以一个具体问题为例,如何监测哪个IP来源对数据库表statistics_test进行了UPDATE操作,文内探讨了多种方法,并提供了详细的代码... 目录引言1. 为什么需要监控数据库更新操作2. 方法1:启用数据库审计日志(1)mysql/mariad

postgresql数据库基本操作及命令详解

《postgresql数据库基本操作及命令详解》本文介绍了PostgreSQL数据库的基础操作,包括连接、创建、查看数据库,表的增删改查、索引管理、备份恢复及退出命令,适用于数据库管理和开发实践,感兴... 目录1. 连接 PostgreSQL 数据库2. 创建数据库3. 查看当前数据库4. 查看所有数据库

从入门到精通MySQL 数据库索引(实战案例)

《从入门到精通MySQL数据库索引(实战案例)》索引是数据库的目录,提升查询速度,主要类型包括BTree、Hash、全文、空间索引,需根据场景选择,建议用于高频查询、关联字段、排序等,避免重复率高或... 目录一、索引是什么?能干嘛?核心作用:二、索引的 4 种主要类型(附通俗例子)1. BTree 索引(

Oracle 数据库数据操作如何精通 INSERT, UPDATE, DELETE

《Oracle数据库数据操作如何精通INSERT,UPDATE,DELETE》在Oracle数据库中,对表内数据进行增加、修改和删除操作是通过数据操作语言来完成的,下面给大家介绍Oracle数... 目录思维导图一、插入数据 (INSERT)1.1 插入单行数据,指定所有列的值语法:1.2 插入单行数据,指