您当前的位置: 首页 > 

测试萌萌

暂无认证

  • 3浏览

    0关注

    1003博文

    0收益

  • 0浏览

    0点赞

    0打赏

    0留言

私信
关注
热门博文

谈谈2020,最新Model的设计-APP重构之路

测试萌萌 发布时间:2020-10-27 16:25:17 ,浏览量:3

很多的app使用MVC设计模式来将“用户交互”与“数据和逻辑”分开,而model其中一个重要作用就是持久化。下文中设计的Model可能不是一个完美的,扩展性强的model范例,但在我需要重构的app中,这样的设计能够满足我的需要。

关于Model

Model层包含了app的数据与逻辑,Model层中的类需要关心的是数据的表现,存储,以及操作。Model层是整个app生态中相对独立的一个部分,因为它不会直接与controller层或者是View层进行通讯,而是在其他层需要请求它的信息的时候进行间接通讯。

Model有什么用?

想要写好一个model,首先要清晰Model的作用。

  • 属性存取:将文件中的一些特性和数据以属性的形式存储
  • 可变性:属性可以readwrite,所以能够被改变,并保存到本地
  • KVO:可以观察一个属性的值并在它改变的时候受到通知,并以此对UI或其他地方进行控制
  • 处理数据:根据业务逻辑处理网络获取数据与本地存储数据
如何定义Model类

我们可以创建一系列的Model类,它们之间可以互相继承,同时每一个Model类是与当前app中的实体对应。比如在我当前需要重构的app中,用户数据对应的是UserInformationModel,班级信息对应的是StudentClassModel。

另一方面,在Model类的实现过程中,有许多问题需要解决,所以我下面会根据我当前正在重构的app的一些情况结合来解释。

信息的存储格式处理

数据可以以各种不同的格式储存,在我重构的app中,数据等信息是使用普通的数据结构来存放,比如使用数据或者是字典来保存Model的信息。在一开始建立Model的时候并没有太大的问题,但是当需求不断增加,一个Model类的信息开始变得庞大起来的时候,问题就开始浮现了。比如下面我要输出用户的姓名、年龄、班级、班主任、年级等数据。

// never do this !!!
- (void)printInformation
{
    NSLog(@"name: %@", [user objectForKey:@"name"]);
    NSLog(@"age: %@"m [user objectForKey:@"age"]);
    NSLog(@"class: %@"m [user objectForKey:@"clazz"]);
    NSLog(@"teacher: %@"m [user objectForKey:@"teacher"]);
    NSLog(@"grade: %@"m [user objectForKey:@"grade"]);
}

这样取出数据似乎没什么问题,但是当数据多起来的时候就会发现代码会十分混乱,而且不!美!观!,同时最主要的问题是,在从字典中取出数据的过程中会把key打错,导致数据取出失败。

所以在设计Model过程中,尽可能将Model设计成一个类而不是一个结构,在类中可以使用属性(Property)来存取信息,它能够提供给开发者基本的拼写检查,完全杜绝打错key导致的数据获取失败的情况,同时方便其他开发者看到Model中存储的数据类型,也方便日后的扩展与维护。

// YES! do this!
- (void)printInformation
{
    NSLog(@"name: %@", user.name);
    NSLog(@"age: %@", user.age);
    NSLog(@"class: %@", user.clazz);
    NSLog(@"teacher: %@", user.teacher);
    NSLog(@"grade: %@", user.grade);
}
网络数据处理

由于我重构的app大部分功能是基于网络的,所以网络数据在Model层的处理会是重点。因为网络数据是异步获取的,而且获取过程很容易失败。所以网络数据的获取比本地数据的获取难度要更大。另一方面,很多app的网络请求框架中都提供了缓存功能,可以在下一次请求中从缓存中更快的获取数据,而不需要再次进行网络请求,这里又涉及到本地数据获取等问题,使得网络数据的处理显得尤其繁琐。

在一个同步的网络环境下,我们可以把错误处理放到其他地方,可以简单的做缓存,甚至可以像处理本地数据一样来更新、删除、添加新的网络数据。但很不幸的是,网络是异步的,所以我们需要处理这个重要问题。

首先我在上文提到的缓存、失败处理,这些应该是交给网络请求框架来进行处理,最后得到一个responseObject,再对responseObject进行model转换等操作。

比如我在当前的项目中,是利用了MJExtension来进行字典与模型之间的转换,而这个转换,我是放到了每个单独的API中,比如对于库存的请求,我在InventoryAPI中进行对responseObject转换成InventoryModel的操作,而不是放到vc中进行,这样在vc中仅仅是进行网络请求,请求完成后返回的是我想要的model。

InventoryAPI.m

- (id)modelingFormJSONResponseObject:(id)JSONResponseObject
{
    NSUInteger count = ((NSArray *)JSONResponseObject).count;
    NSMutableArray *modelsArray = [NSMutableArray array];
    for(int i = 0; i             
关注
打赏
1663571372
查看更多评论
0.1049s