西柚买手app编写的技术文档

前言

上家公司已经倒闭重组在当时技术会议上时负责编写的技术文档,技术文档所涉及的东西都是当时接手项目以后修改的架构(16年中下旬),在现在来看有很多还不够成熟的地方也有很多设计依然是非常规范的,在后面的博客会慢慢讨论,希望本文能够给您的移动项目演进提供一些借鉴。

架构设计

客户端功能基础流程图.

    ---------------     ---------------     ---------------     ---------------
|             |     |             |     |             |     |             |
| 调用网络API  | --> |   展现列表    | --> |  选择列表    | --> |   展现单页   |
|             |     |             |     |             |     |             |
---------------     ---------------     ---------------     ---------------
                           ^                                        |
                           |                                        |
                           |                                        |
                           ------------------------------------------

架构分层

客户端使用的是四层架构方式
1.展现层:可重用功能性控件的封装及ViewController中的UI处理
2.业务层:弱业务逻辑结构的主要存放地及model层
3.网络层:统一的网络API调用及回调方式
4.本地数据层:缓存API的封装

关于View的布局

使用Autolayout布局:主要使用两大Autolayout框架
1.Masonry:成熟,使用人数多
2.SDAutolayout:国人开发,结构读写简介便利
可以根据自己的开发习惯挑选.

关于storyboard,nib,code

1.storybard禁止使用,在多人开发的场景下,容易造成难以解决的冲突.
2.大部分的场景比如:拥有复杂界面元素,核心动画需使用code去画view
3.简易View可以使用nib直接拖拽比如简易的cell或者某个悬浮控件.

注意事项:尽少使用统一派生的ViewController.

关于胖Model和瘦Model

在客户端中涉及到网络API中的Model大部分使用瘦Model业务逻辑并没有在Model层中处理,而是仅仅做了对于数据的处理.而在其他方面涉及到弱业务逻辑的时候,使用的是胖Model对比如下

 Raw Data:
     timestamp: 1234567
 FatModel:
     @property(nonatomic,assign)CGFloat timestamp;
     - (NSString *)ymdDateString; //2015-04-20 15:16
     - (NSString *)gapString; //3分钟前、一小时前、一天前、2015-3-13 12:34
 Controller:
     self.dateLabel.text = [FatModel ymdDateString];
     self.gapLabel.text = [FatModel gapString];
 
 Raw Date:
     Dic: JSONDic
 ThinModel:
     #property(nonatomic,assign)NSString *name;
     #property(nonatomic,assign)Nsstring *age;
 Controller:
     ThinModel *model = [ThinModel MJExtent:Dic];
     self.nameLabel.text = model.name;
     self.ageLabel.text = model.age;

关于ViewController中业务逻辑处理规范

viewController代码规范

@property(nonatiomic,strong)UIButton *confirmButton  
...

#pragma mark --life cycle
viewDidLoad
viewWillAppear
...

#pragma mark - UITableViewDelegate
methods
...
#pragma mark - CustomDelegate
methods
...

#pragma mark - event response
- (void)didTappedConfirmButton:(UIButton *)confirmButtom
...

#pragma mark - private methods
methods
...

#pragma mark - getters and setters
- (UIButton *)confirmButton
- (UITableView *)tableView
...

注意事项:所有的属性都是用getter和setter.在viewDidload里面只做addSubview的事情.

iOS客户端统一使用XYNetWorkHeapler的API.

在于业务层对接的部分API统一使用的Delegate回调方式限制灵活性,以此来交换应用的可维护性.但是在API的内部还是使用了Block的形式回调网络请求保存灵活性。

     [AFNetworkingAPI callApiWithParam:self.param successed:^(Response *response){
        if ([self.delegate respondsToSelector:@selector(successWithResponse:)]) {
            [self.delegate successedWithResponse:response];
    }
} failed:^(Request *request, NSError *error){
        if ([self.delegate respondsToSelector:@selector(failedWithResponse:)]) {
            [self failedWithRequest:request error:error];
    }
}];

关于交付什么样的数据给业务层

API交付的是转换后的字典而且仅在服务器返回的code合法(为0)的情况下才会拥有回调数据.其他情况下会回调nil在viewController中做出相应的处理.并不要直接回调对应的模型,以便为以后业务的拓展做铺垫.

在业务层使用的是离散型API调用方式

离散型API调用方式.一个API对应于一个APIManager,然后这个APIManager只需要提供参数就能起飞,API名字、着陆方式都已经集成入APIManager中,例如

 离散型API调用方式:

 @property (nonatomic, strong) ItemListAPIManager *itemListAPIManager;

// getter
- (ItemListAPIManager *)itemListAPIManager
{
if (_itemListAPIManager == nil) {
    _itemListAPIManager = [[ItemListAPIManager alloc] init];
    _itemListAPIManager.delegate = self;
}

return _itemListAPIManager;
}

// 使用的时候就这么写:
[self.itemListAPIManager loadDataWithParams:params];

关于API新增接口方法命名

好的函数名:
- (NSDictionary *)exifDataOfImage:(UIImage *)image atIndexPath:(NSIndexPath *)indexPath;

坏的函数名:
- (id)exifData:(UIImage *)image position:(id)indexPath callback:(id<ErrorDelegate>)delegate;

为什么坏?
1. 不要直接返回id或者传入id.
2. 要告知业务方要传的东西是什么,比如要传Image,那就写上ofImage。如果要传位置,那就要写上IndexPath,而不是用position这么笼统的东西.
3. 没有任何理由要把delegate作为参数传进去,一定不会有任何情况不得不这么做的。而且delegate这个参数根本不是这个函数要解决的问题的充要条件.

在关于数据库操作方面iOS客户端并没有使用苹果自带的Core Data,而是使用了iOS领域比较常用FMDB.数据库方案主要是为了便于增删改查,当数据有状态和类别的时候最好还是采用数据库方案比较好,而且尤其是当这些状态和类别都是强业务相关的时候.

FMDBAPI

iOS客户端在关于数据库操作的时候统一调用XYFMDBManagerAPI XYFMDBManagerAPI已封装好关于创建新表及增删改查的方法.例如

   #pragma mark 群组数据库处理
   /**根据群组唯一标示创建一个数据库表*/
   - (void)CreatGroupNewTable:(NSString *)TableName;

   /**根据群组标示向标中插入数据*/
   - (void)AddDataImageUrl:(NSString *)TableName WithImageUrl:(NSString *)imageUrl;

   /**根据群组标示查询表中所有数据*/
   - (NSArray *)QueryData:(NSString *)TableName;

缓存淘汰算法:
1.使用LRU(least-recently-used) 算法来淘汰使用频率较低的缓存。
2.不同的缓存给予最长缓存清除时间。

持久层与业务层的交互方式

             -------------------------------------------
             |                                         |
             |  LogicA     LogicB            LogicC    |    ------------------------------->    View Layer
             |     \         /                 |       |
             -------\-------/------------------|--------
                     \     /                   |
                      \   / Virtual            | Virtual
                       \ /  Record             | Record
                        |                      |
             -----------|----------------------|--------
             |          |                      |       |
Strong Logics  |     DataCenterA            DataCenterB  |
             |        /   \                    |       |
-----------------|-------/-----\-------------------|-------|       Data Logic Layer   ---
             |      /       \                  |       |                         |
Weak Logics   | Table1       Table2           Table     |                         |
             |      \       /                  |       |                         |
             --------\-----/-------------------|--------                         |
                      \   /                    |                                 |--> Data Persistance Layer
                       \ / Query Command       | Query Command                   |
                        |                      |                                 |
             -----------|----------------------|--------                         |
             |          |                      |       |                         |
             |          |                      |       |                         |
             |      DatabaseA              DatabaseB   |  Data Operation Layer ---
             |                                         |
             |             Database Pool               |
             -------------------------------------------

持久层有专门负责对接View层模块或业务的DataCenter,它们之间通过Record来进行交互。DataCenter向上层提供业务友好的接口,这一般都是强业务:比如根据用户筛选条件返回符合要求的数据等。

然后DataCenter在这个接口里面调度各个Table,做一系列的业务逻辑,最终生成record对象,交付给View层业务。

DataCenter为了要完成View层交付的任务,会涉及数据组装和跨表的数据操作。数据组装因为View层要求的不同而不同,因此是强业务。跨表数据操作本质上就是各单表数据操作的组合,DataCenter负责调度这些单表数据操作从而获得想要的基础数据用于组装。那么,这时候单表的数据操作就属于弱业务,这些弱业务就由Table映射对象来完成。

Table对象通过QueryCommand来生成相应的SQL语句,并交付给数据库引擎去查询获得数据,然后交付给DataCenter。

其他本地持久化方案使用场景规范

NSUserDeault
一般来说,小规模数据,弱业务相关数据,都可以放到NSUserDefault里面,内容比较多的数据,强业务相关的数据就不太适合NSUserDefault了。

keychain
Keychain是苹果提供的带有可逆加密的存储机制,普遍用在各种存密码的需求上。另外,由于App卸载只要系统不重装,Keychain中的数据依旧能够得到保留,以及可被iCloud同步的特性,大家都会在这里存储用户唯一标识串。所以有需要加密、需要存iCloud的敏感小数据,一般都会放在Keychain。

文件存储
文件存储包括了Plist、archive、Stream等方式,一般结构化的数据或者需要方便查询的数据,都会以Plist的方式去持久化。Archive方式适合存储平时不太经常使用但很大量的数据,或者读取之后希望直接对象化的数据,因为Archive会将对象及其对象关系序列化,以至于读取数据的时候需要Decode很花时间,Decode的过程可以是解压,也可以是对象化,这个可以根据具体<NSCoding>中的实现来决定。Stream就是一般的文件存储了,一般用来存存图片啊啥的,适用于比较经常使用,然而数据量又不算非常大的那种。

注意事项与各项指标及改进方向

内存泄漏

在iOS客户端最初版本中,因为内存泄漏所导致的error非常的多.所以关于内存泄漏方面是现如今每一次迭代开发所注意的重中之重。内存泄漏注意事项:
1.block导致的循环引用
2.timer是否持有self
3.声明delegate为strong类型
4.CoreFoundation对象(C对象)

Crash

目前使用的Crash统计工具为bugly,Crash率的指标定为千分之三.

改进方向

目前版本针对改进方向,网络层的继续优化及冗余代码的重构.

FAQ

在使用网络API的时候自己新增方法代理回调程序Crash

请检查是否对代理是否还拥有着路点可以回调进行了判断

  if ([self.delegate respondsToSelector:@selector(successWithResponse:)]) {
            [self.delegate successedWithResponse:response];
    }

项目Clone之后关于swift的类报错

请检查Xcode版本,iOS客户端部分模块使用的swift2.0进行编写,需要在Xcode7.3以上才可以支持swift2.0的语法。

无xcworkspace文件无法启动工程

如还未安装cocoapods请使用终端下载安装,因为所有的第三方库均托管给cocoapods进行托管.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352