Swift开发架构及注意事项

一 : 简单介绍以下几个方面

1 编码规范
2 设计模式的选择
3 项目目录结构
4 网络层
5 数据存储
6 日志收集
7 安全性
8 测试(功能,性能等)和自动化打包(比如持续集成管理工具)

二 : 项目一般满足点

  • 代码整齐,分类明确 ,没有common,没有core
  • 多聚合 , 少继承 , 多协议;高内聚 , 低耦合
(1) 功能内聚和数据耦合,是我们需要达成的目标。
(2) 横向的内聚和耦合,通常体现在系统的各个模块、类之间的关系,
    而纵向的耦合,体现在系统的各个层次之间的关系。
(3) 少继承, 多考虑协议
(4) 最终: 易测试,易拓展
  • 思路和方法要统一,尽量不要多
  • 对公用方法该限制的地方有限制,该灵活的灵活
  • 保持一定量的超前性
  • 接口少,接口参数少, 方法语义明确
  • 高性能

三 : 编码规范

详见: Swift编码规范

补充:

1 避免用OC式的语法写Swift : 举例赋值字符串操作,为空设置默认值

  • OC式的语法写Swift
 var nameString  : String
 var newNameString : String?
 newNameString = nil

 if (newNameString != nil) {//不为空赋值
      nameString = newNameString!
 } else {
     nameString = "昵称"
 }
  • Swift语法写法:
if let nameStr = newNameString { //解包成功赋值
    nameString = nameStr
} else {
     nameString = "昵称"
}
  • 简洁优雅的使用Swift:
nameString = newNameString ?? "昵称"

2 充分利用Swift的新特性
能用协议的尽量, 少用继承, 这样有利于项目扩展和分工合作等等

3 个人习惯,能用code少用storyboard 和 xib

当团队具有一定规模的(比如iOS开发 > 5 人),storyboard 和 xib缺点和明显
同一份代码文件的作者会有很多,不同作者同时修改同一份代码的情况也不少见。
因此,使用Git进行代码版本管理时出现Conflict的几率也比较大。
需求变化非常频繁,为了完成需求而针对现有代码进行微调的情况,以及针对现有代码的部分复用的情况
复杂界面元素、复杂动画场景的开发任务

4代码注释

/**
 函数描述信息,支持换行(空一行作为换行符)
 
 描述支持Markdown语言:
 
 - 信息1
 - 信息2
 - 信息3
 
 1. item 1
 2. item 2
 3. item 3
 
(下面的'/'实际代码中去掉)
 /```
 let a = "Swift"
 let b = "代码注释"
 print( a + ", " + b )
 /```
 
 */
func markdownMsg() {}


/// 支持常用类型描述,比如参数 **Parameter**, 错误信息 **Throws** 和返回信息 **Returns**.
///
/// - Parameters:
///   - Param1: 参数1
///   - Param2: 参数2
///
/// - Throws: 条件不满足时返回的错误类型
///
/// - Returns: 正常执行返回的结果
func parameterMsg(Param1: AnyObject?, Param2: AnyObject?) throws -> String {
    if Param1 == nil && Param2 == nil{
        throw MsgError.BothNilError
    }
    return "result"
}
enum MsgError: Error{
    case BothNilError
}


按住option点击方法:(还有其他注释如 MARK 等)


图片.png
图片.png

四 : 设计模式的选择 : MVVM ( MVC、MVCS、VIPER )

目前iOS常用的几种设计模式:代理模式、观察者模式、单例模式、策略模式、工厂模式、MVC模式等

MVC模式
当一个项目越来越大的时候,纯粹的MVC模式会变得越来越难维护了,所以有必要对其进行改进
一般项目可以考虑使用MVVM

View <-> Controller <-> ViewModel <-> Model

附: MVVM模式配合使用框架RXSwift 和RXcocoa开发,代码会更简洁

RX学习资料:航歌hangge.com

1 简单聊点 依赖 问题:

从A页面push到B页面:

mineVC.title = "个人主页"
mineVC.ID = personID
mineVC.isMe = isMe
self!.navigationController?.pushViewController(mineVC, animated: true)

如果以后需求改变了,需要增加几个参数或者是删除参数, 如此改动会很麻烦
那么这些操作其实不必要Controller来做:

PushModel : 专门用来管理需要被push或者present事件
BDataModel : 专门声明需要传递的参数,根据需要进行扩展

如此,控制器和控制器之间不会过度依赖,只需要告诉PushModel我有个BDataModel给你就行

2 简单聊聊高频使用的UITableViewCell,通过cellForRowAt方法如何瘦身来理解如何拆分

(1) 瘦身前的控制器: 显得异常臃肿

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let aNews = news[indexPath.row]
        switch aNews.cell_type {
        case .user:             // 用户
            let cell = tableView.ym_dequeueReusableCell(indexPath: indexPath) as HomeUserCell
            cell.readCountLabel.text = "\(aNews.readCount)阅读"
            cell.verifiedContentLabel.text = aNews.verified_content
            cell.digButton.setTitle(aNews.diggCount, for: .normal)
            cell.commentButton.setTitle("\(aNews.commentCount)", for: .normal)
            cell.feedshareButton.setTitle(aNews.forwardCount, for: .normal)
            cell.contentLabel.attributedText = aNews.attributedContent
            //cell.aNews = aNews
            return cell
        case .relatedConcern:   // 相关关注
            let cell = tableView.ym_dequeueReusableCell(indexPath: indexPath) as TheyAlsoUseCell
            cell.readCountLabel.text = "\(aNews.readCount)阅读"
            cell.verifiedContentLabel.text = aNews.verified_content
            cell.digButton.setTitle(aNews.diggCount, for: .normal)
            cell.commentButton.setTitle("\(aNews.commentCount)", for: .normal)
            cell.feedshareButton.setTitle(aNews.forwardCount, for: .normal)
            cell.contentLabel.attributedText = aNews.attributedContent
            cell.theyUse = aNews.raw_data
            return cell
        }
    }

(2) 瘦身后的控制器 : 通过view的set方法,把赋值代码移动到cell里,或者建立一个DataModel来处理这件事情.

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let aNews = news[indexPath.row]
        switch aNews.cell_type {
        case .user:             // 用户
            let cell = tableView.ym_dequeueReusableCell(indexPath: indexPath) as HomeUserCell
            cell.aNews = aNews
            return cell
        case .relatedConcern:   // 相关关注
            let cell = tableView.ym_dequeueReusableCell(indexPath: indexPath) as TheyAlsoUseCell
            cell.theyUse = aNews.raw_data
            return cell
        }
    }

上面代码仍然有一定的冗余,两个cell进行了两次初始化,如果十个呢,显然会越来越臃肿.

(3) 进一步瘦身后的控制器 : 控制器只需要对cell进行一次赋值就行,具体什么cell让cell的DataModel去做就行了.

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let row = fromData[indexPath.row]
        let cell = tableView.dequeueReusableCell(withIdentifier: row.reuseIdentifier, for: indexPath)
        row.update(cell: cell)
        return cell
    }
3 简单聊聊Swift协议编程 [ 协议的延展以及和结构体配合 ]
  • Swift is a Protocol-Oriented Programming Language( Swift 是一门面向协议 (POP) 开发的语言)

  • Swift 的核心是面向协议编程

  • 面向对象与面向协议比较:**

面向对象是一个很传统的软件开发模式,通过类来实现
面向协议是苹果在 swift 中主推的,通过协议和结构体,可以代替绝大部分的类,面向协议使代码更加灵活,类似于组件化开发,符合工厂方法模式

  • Swift 中的很多对象都改成了结构体和协议

Classes : 6
Enum : 8
Struct : 103
Protocol : 86

  • 回顾一下OC的单继承:

1 子类是通过继承父类获得父类的方法
2 通过继承添加的方法,子类不一定每个都会用,使代码冗余
3 拥有太多的子类, 类冗余
4 过度依赖父类的方法,当父类的方法更改后,影响子类的重载方法
5 不能多继承

  • Swift协议编程的几个点

1 想要添加一些属性和方法,给需要的地方, 组件化
2 为协议可以添加属性 和 方法 protocol xxxProtocol{}
3 接受协议的结构体 struct xxxModel: xxxProtocol {}
4 对协议进行扩展 extension xxxProtocol {}
5 协议的协议 protocol xxxProtocol xxxProtocol{}

  • 协议的使用 :类,协议1,协议2,协议3, .. {}
class ViewController: UIViewController, AProtocol, BProtocol, CProtocol {...}
4 简单聊聊扩展 extension

1 可以协议,自定义对类,系统类 等等 进行属性和方法延展
比如对系统类UIViewController添加属性和方法
2 如果类和协议等进行了扩展,那么子类或者协议遵循者就自动拥有该扩展功能
3 扩展一般是基础属性和功能,比如class动物,那么一定拥有eat的功能,但是不一定有fly(那么fly就可使用协议)
4 对同一个类扩展相同的方法和属性,那么就会导致冲突,可以使用命名空间解决

extension UIViewController {
    private struct YXVCKey {
        static var sKey = "dataSourceKey"
    }
    //为系统类添加属性
    public var dataSource : Array<Any> {
        get {return (objc_getAssociatedObject(self, &YXVCKey.sKey) as? Array<Any>)! }
        set { objc_setAssociatedObject(self, &YXVCKey.sKey, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC) }
    }
    
    //为类添加方法
    public func changeViewColor(_ bgColor :UIColor) -> () {
        self.view.backgroundColor = bgColor
    }

}

五 : 项目目录结构 : 根据不同项目需求决定

  *  Base : 存放一些基类,比如BaseViewController,BaseModel等,共性直接在基类中去修改
  *  Vendor : 三方,因为我的项目中使用cocopods管理三方,所以这个文件夹中我在此放的是一些比较小的功能的第三方或者不支持cocopods的
  *  Framework : 存放一些类库或者自己封装的一些静态库
  *  Resource : 存放app中一些索引资源,比如图片,文本等,或者将图片打包的Bundle
  *  Custom : 这个文件夹我用来存放自己项目或者公司自己风格的一些自定义的视图框架( 或SDK )
  *  Network : 这个只专门用来做网络处理的,因为这个项目基本上都会用到网络请求,算是比较重要的一个部分,所以在此单独拿出来作为一个分类
  *  Expand : 扩展文件
      -- Tool : 工具
      -- DataBase: 数据存储相关
      -- Macros : 宏定义
      -- Const : 常量
  *  Main : AppDelegate或者AppDelegate的Category 等等
  *  Class : 存放的是App中所有的模块功能
       - Public : 公用
       - Home : 比如首页
             -- Controller
             -- Model
             -- View
             -- ViewModel
             -- Other

 * XXX-Tests :        单元测试
 * XXX-UITests  :   UI测试
 * Products :         系统自动生成的.app所在文件夹
 * Pods  :               Pods第三方库

六 : 网络层

Alamofire : 首选主流网络开源框架
Moya
RXSwift / RXCocoa

pod "Moya/RxSwift"

以下是艺下NetWork.swift 示例代码:
import Foundation
import Moya
public enum NetWork {
    case login(mobile:String,captcha:String)
}

extension NetWork:TargetType{
    public var sampleData: Data {
        return Data.init()
    }
    
    /// The target's base `URL`.
    public var baseURL: URL {
        return API.base.rawValue.url!
    }
    
    // The path to be appended to `baseURL` to form the full `URL`.
    public var path: String {
        switch self {
        case .login(mobile: _, captcha: _):
            return "/auth/login";
        }
    }
    
    // The HTTP method used in the request.
    public var method: Moya.Method {
        switch self {
        case .login(mobile: _, captcha: _):
            return .get;
        }
    }

    // The type of HTTP task to be performed.
    public var task: Task {return .requestPlain}

    // The type of validation to perform on the request. Default is `.none`.
    public var validationType: ValidationType { return .none }
    
    // The headers to be used in the request.
    public var headers: [String: String]? {
        var AlamHeaders = [*****]
        AlamHeaders["sign"] = ***
        (一堆设置)
        return AlamHeaders
    }
}

七 : 数据存储
根据需求决定持久化方案
持久层与业务层之间的隔离
持久层与业务层的交互方式
数据迁移方案
数据同步方案
损坏修复
等等
目前移动端数据库方案按其实现可分为两类,

一 : 关系型数据库,代表有CoreData、FMDB等。
CoreData
它是苹果内建框架,和Xcode深度结合,可以很方便进行ORM;但其上手学习成本较高,不容易掌握。稳定性也堪忧,很容易crash;多线程的支持也比较鸡肋。
SQLite
FMDB
它基于SQLite封装,对于有SQLite和ObjC基础的开发者来说,简单易懂,可以直接上手;而缺点也正是在此,FMDB只是将SQLite的C接口封装成了ObjC接口,没有做太多别的优化,即所谓的胶水代码(Glue Code)。使用过程需要用大量的代码拼接SQL、拼装Object,并不方便。

二 : key-value数据库,代表有Realm、LevelDB、RocksDB等。
Realm
1 因其在各平台封装、优化的优势,比较受移动开发者的欢迎。对于iOS开发者,key-value的实现直接易懂,可以像使用NSDictionary一样使用Realm。并且ORM彻底,省去了拼装Object的过程。
2 是一个跨平台的移动数据库: Java,Objective-C,Swift,React Native,Xamarin 等等
3 Realm支持事务,满足ACID:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。

三 : WCDB 是微信推出的一个高效、完整、易用的移动数据库框架,基于SQLCipher,支持iOS, macOS和Android。易用,支持事务,可加密、损坏修复。WCDB在迭代中也在慢慢的稳定

Realm、WCDB与SQLite移动数据库性能对比测试:

image
image
目前比较好的是Realm 和 微信的 WCDB ,任意一个都可以,可根据项目自己选择 .

附:Realm的简单使用

//创建数据管理对象
let realm = try! Realm(configuration: Realm.Configuration(inMemoryIdentifier: "TemporaryRealm"))
// 数据持久化操作(类型记录也会自动添加的)
try! realm.write {
realm.add(self.userInfo)
}
//数据库地址
print(realm.configuration.fileURL ?? "")

//获取模型数据
var consumeItems:Results<UserInfoModel>?
consumeItems = realm.objects(UserInfoModel.self)
let testType1 = consumeItems?.first
print(testType1?.data?.refreshToken ?? "failue")

八 : 日志收集

一个 APP不可避免的要对其进行,bug收集和日志分析

1 一种方式是发生崩溃以后上传日志到自己的服务器,然后进行分析
XCGLogger

2 另外一种是第三方接口, 比如bugly的日志收集管理系统 : 功能比较强大,不仅仅是崩溃日志,甚至是卡顿等都能收集,并且有可视化的管理系统方便查看和分析

根据业务需要选择一个合适的方式.

九 : 安全性

  1. 对修改请求进行加密认证,修改使用POST ,DELETE 等 提交表单
    展示和分享相关页面使用GET请求,方便分享,收藏等等
  2. 有条件使用https
  3. 对用户敏感数据 使用密钥进行加密
  4. 本地配置文件中的敏感数据进行加密 或 保存在keychain中
    等等

十 : 自动化打包和测试,持续集成工具

测试是对项目健全性能的保障:包括 性能、内存weak、UnitTest,AutoTest 、UITest 等等

  • UnitTest 单元测试 分为3种:
    1 逻辑测试:测试逻辑方法. --逻辑测试
    2 异步测试:测试耗时方法(用来测试包含多线程的方法): --异步测试
    3 性能测试:测试某一方法运行所消耗的时间 --性能测试

  • UI 自动化测试 :
    iOS 自动化测试 UI Automation 【xcode 8 找不到 Automation ?】

  • 代码版本管理 : 持续集成工具
    例如 CruiseControL,hudson ,jenkins,还有apache的Continuum 等 开源的持续集成工具
    jenkins 配置

十一 : 几点注意事项

  • 注意内存泄漏的情况: 比如循环强引用,闭包等
  • 适配常规机型
  • 本地低频使用的大图尽量不要使用imageName加载到缓存
  • 图片加载保存流程图 :


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

推荐阅读更多精彩内容