【iOS】小项目框架设计(ReactiveCocoa+MVVM+AFNetworking+FMDB)

网址: http://www.saitjr.com/ios/ios-framework-reactivecocoa-mvvmafnetworking-fmdb.html

 

 

 

上一个项目使用到了ReactiveCocoa+MVVM+AFNetworking+FMDB框架设计,从最初的尝试,到后来不断思考和学习,现在对这样一个整体设计还是有了一定了理解与心得。在此与大家分享下。

本文将不再过多的描述ReactiveCocoaMVVMFMDB的使用细节。关于ReactiveCocoa,我有一篇实用案例的博客:

http://www.brighttj.com/ios/ios-reactivecocoa-utility-demo.html

文章介绍的更多的是我对这个框架设计的理解,而不是具体代码逻辑的讲解。关于代码逻辑,我会在Demo中给出详细的注释,本文Demo下载地址:

https://github.com/saitjr/ReactiveCocoa-MVVM-AFNetworking-FMDB.git

环境信息:

Mac OS X 10.11

Xcode 7.0.1

iOS 9.0.1

ReactiveCocoa 2.4.7

AFNetworking 2.6.1

FMDB 2.5

MJExtension 2.5.14


正文

工程目录

先来谈谈工程目录吧,如图:

工程目录

工程目录

1. 【Module】+【Model】

在这个目录中,比较核心的是【Module】与【Model】,他们组成了整个MVVM框架。

【Module】与【Model】均包含【Base】,其中有BaseModelBaseViewModelBaseViewController。在开发中,我还是习惯无论是否需要基类,都去写一个。难免开发之初就考虑到,也难免之后需求会变更。

2. 【Interface】接口

这是借鉴了Java中的接口思想,目的是为了统一方法名。例如里面的SQLInterface.h文件,就是一个对数据进行CRUD操作的protocol,并且可以规定里面的方法是否必须实现。

3. 【Configuration】配置

对项目的一些基本配置,如基本宏定义、常量、通知名,亦或是Cell的identifier。宏定义中一般包含项目基本属性,如:主色调、常用方法等。

在提供的Demo中,我将SQL语句放在了SQL.h中,是因为SQL只有一个文件在引用,其中的定义方式是:

static NSString * const selectArticleSQL = @"SELECT * FROM article";

NotificationNames.h会在大部分文件中用到,所以使用UIKIT_EXTERN定义为了全局变量:

-----.h
UIKIT_EXTERN NSString * const LoadAllNotification;
-----.m
NSString * const LoadAllNotification = @"LoadAllNotification"; 

5. 【Category】类目

项目中没有打包的类目,例如给三方或系统类新增的一些方法。

RAC+MVVM

RAC+MVVM在【Module】和【Model】这两个目录中进行实现。在这之中,MVVM是框架思想,RAC只是辅助而已。

一、MVVM

之所以采用MVVM,而不是MVC,也得益于MVVM的一大特点,就是减轻C层的负担,毕竟以前的C层完全就是百宝箱,什么样的代码都写在里面。

对于MVVM我的理解和拆分是这样的:

1. Model与View

这一层和MVC中的Model、View含义相同。

2. ViewModel

这一层主要作用是将以前写在ViewController中的数据处理放在ViewModel中,如:网络请求、数据缓存、无法直接展示的数据处理(如NSNumber这类的,就在VM中处理成NSString,然后V层直接用,而不是在V层中处理)。

二、Demo中VC的设计

图形结合源码应该能看个大概。

VC设计

VC设计

三、自定义cell的设计(可延伸至自定义View的设计)

因为介绍的是VC,所以这里再单独说一下HomePageCell这个自定义cell的设计。

因为想到实际项目中,可能会有比较复杂的cell,所以Demo中写的是一个比较完整的设计方式(如果单单看这个Demo的话,这个自定义cell太简单,没必要有一个单独的VM,有点过度设计)。

cell中的构思是,cell有一个CellVM来管理cell中要显示的数据,CellVM来自于VC中,dataSource数组。处理方式具体是:在网络请求完成以后,将字典->模型,然后通过模型,初始化CellVM,然后将CellVM放入dataSource数组。

Cell实现部分代码如下:

@implementation HomePageCell

- (void)awakeFromNib {
    [super awakeFromNib]; [self setupSignal]; } - (void)setupSignal { @weakify(self); [RACObserve(self, viewModel) subscribeNext:^(HomePageCellViewModel *viewModel) { @strongify(self); self.textLabel.text = viewModel.titleText; self.detailTextLabel.text = viewModel.authorText; }]; } @end 

AFNetworking

前几天,AFNetworking升级到了3.0。将以前基于NSURLConnection的API,全都改成了NSURLSession,关于更新的详情,可以看AFNetworking 3.0迁移指南这篇文章。

在页面销毁、重新请求等情况下,需要将还在队列中的请求取消,以免占用资源。

考虑到这一点,结合我的网络请求在VM发送,权衡再三,在BaseViewModel的基础上,再进行了一次封装——RequestViewModel。这个VM有AFHTTPSessionManager类的属性,一个该属性的懒加载和一个在dealloc中取消请求方法。

在以前使用MVC的时候,我会对AFNetworking进行再次的封装,这样更像是一个MVCS的设计,目的是防止VC过重,现在把这部分代码扔在了VM中,看起来还好,所以并没有对AFNetworking再次封装。关于以前的设计方式,可以看这篇文章:

网络请求框架封装

FMDB

FMDB提供了一种线程安全的模式,在这之中维护这一个串行队列。

1. 初始化

初始化方式参考了开源项目MVVMReactiveCocoa,作者采用了类目的形式给了一个单例:

@implementation FMDatabaseQueue (Extension)

+ (instancetype)shareInstense {
    
    static FMDatabaseQueue *queue = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ queue = [FMDatabaseQueue databaseQueueWithPath:DB_PATH]; }); return queue; } @end 

2. 统一接口

关于CRUD方法的定义,借鉴了Java Interface的设计:

3. SQL语句

根据实际情况,SQL语句参数可以采用?的形式,也可以采用:keyword的形式。

INSERT INTO myTable VALUES (?, ?, ?)
INSERT INTO myTable VALUES (:id, :name, :value)

关于数据库的创建与更新,我并不建议在Bundle中包含xxx.sql这样的文件,因为他们会在编译后一起打包,用户下载解压就能看到,并不怎么安全。目前我是直接在程序中写的SQL,不知道大家有没有更好的方式。

4. CRUD

在网络请求成功的时候,存入数据。存储是一个批量的操作,建议采用事务inTransaction实现。简单的操作采用inDatabase即可。

FMDatabaseQueue是一个串行队列,并且inTransactioninDatabase都是同步线程,需要注意的是不要在block中执行另一个数据库访问操作,防止线程死锁

最后

每个项目完后,都会有很多收获,有很多东西需要整理总结。写这篇博客的原因有两个:

原因之一:因为我在开发过程中踩了不少坑,可能开发到中途,发现框架设计不好。框架如何设计,并没有一个标准答案,而且设计思想,还需要在不断实践中得出,所以每次总结,是为了给自己看,也是为了帮到其他有同样困扰的朋友。

原因之二:也正是因为我不知道框架到底怎么样,所以写出来,让大家看看,都多多提出建议。谢谢。

转载于:https://www.cnblogs.com/DMDD/p/5046003.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Jetpack是一个结合MVVM的快速开发框架,它基于MVVM模式并集成了谷歌官方推荐的Jetpack组件库,包括LiveData、ViewModel、Lifecycle和Navigation组件。这个框架使用Kotlin语言,并添加了大量的拓展函数,以简化代码。它还集成了Retrofit网络请求和协程,可以帮助开发者快速开发项目。\[1\] MVVM是一种软件架构模式,它将应用程序分为三个主要部分:模型(Model)、视图(View)和视图模型(ViewModel)。在MVVM中,视图负责显示数据和用户交互,模型负责处理数据和业务逻辑,而视图模型则充当视图和模型之间的中间层,负责管理视图的状态和数据。Jetpack的MVVM模式可以帮助开发者更好地组织和管理代码,提高开发效率和舒适度。\[1\] 使用Jetpack和MVVM可以带来许多好处,例如简化代码、提高开发效率、提供更好的代码结构和可维护性。通过使用Jetpack的组件库,开发者可以更轻松地处理生命周期管理、数据共享和导航等常见任务。而MVVM模式则可以帮助开发者更好地分离关注点,使代码更易于测试和维护。\[2\] 总之,Jetpack和MVVM是一种强大的组合,可以帮助开发者快速开发Android应用程序,并提供更好的代码结构和可维护性。如果你想了解更多关于Jetpack和MVVM的信息,可以参考引用\[1\]中提供的Jetpack框架的介绍。 #### 引用[.reference_title] - *1* *2* [JetpackMvvm](https://blog.csdn.net/u014608640/article/details/124711159)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [大型Android项目架构:基于组件化+模块化+Kotlin+协程+Flow+Retrofit+Jetpack+MVVM架构实现WanAndroid...](https://blog.csdn.net/m0_37796683/article/details/130277908)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值