我的编程空间,编程开发者的网络收藏夹
学习永远不晚

仓库模式及其在Swift项目中的应用详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

仓库模式及其在Swift项目中的应用详解

正文

在现代 Swift 项目中,很流行一种模式叫做仓库模式,英文是 Repository Pattern。这个模式主要用于构建数据层代码。按照一般的 App 层级划分,一般从上到下划分为 UI 层,业务层,数据层,那么仓库模式的应用位置可以参考下图:

可以看到仓库应用在数据层,业务层通过接口来访问仓库。

不使用仓库模式时的代码

为了研究为什么要使用仓库模式,我们先看看不使用仓库模式时我们是怎么写代码的。一般打开一个界面,会发送网络请求来获取这个界面所需的数据,这时会在 ViewController 写类似下面的代码:

func viewDidLoad() {
    super.viewDidLoad()
    requestData()
}
func requestData() {
    API.request(xxxId: 12345) { result in
        switch result {
        case .success(let model):
            handle(model)
            DispatchQueue.main.async {
                render(model)
            }
        case .failure(let error):
            print(error.localizedString)
        }
    }
}

这种写法在小项目中是没有问题的,但是在稍具规模的项目中,就会对项目的扩展性,维护性,团队合作,开发效率有比较高的要求,这时候就更应该根据科学的软件设计原则来设计更好的架构。

上面的代码在稍具规模的项目中会有以下的缺点:

  • 数据访问代码写在了 ViewController 中,无法测试
  • ViewController 会过于臃肿,难以维护
  • 如果数据访问方式修改了,需要修改 ViewController 中的代码

稍具规模的项目一般会采用 MVVM 等架构,于是以上的代码会写在 ViewModel 中,来避免 ViewController 太过臃肿,但是还是会有无法测试和修改数据访问方式时改动比较大的问题。

使用仓库有什么好处?

总的来说,就是能提供高层次的抽象,从而获得因为抽象带来的一系列好处。

仓库模式提供了数据层的抽象,可以让你的业务代码只依赖一个简单的抽象接口就可以工作。这使得代码松耦合,业务代码不需要知道数据的具体获取和存储细节。

仓库模式让代码可以测试,对于网络请求和数据库读写的部分,可以实现一个提供测试数据的仓库实例,这样就可以编写相应的测试代码。

下面就来看一看怎样使用仓库模式。

设计仓库接口

按照依赖倒转原则,数据访问层作为架构中的一个整体,上层对象在调用数据访问层时应该依赖接口,而不依赖于实现,这样数据访问层的逻辑就可以灵活的变更、替换,比如在网络请求和本地数据之间切换,在不同的网络请求协议之间切换等。

所以设计仓库模式应该先定义接口,定义接口有两种方式,一种是根据具体的数据定义特定的仓库接口,例如对于一个新闻列表的接口:

protocol NewsListRepository {
    func readNewsList() async throws -> [News] 
}

或者是根据不同的业务数据的访问逻辑其实大同小异,可以设计统一的泛型接口:

protocol Repository {
    associatedtype T
    func query(with predicate: NSPredicate?, sortDescriptors: [NSSortDescriptor]?) async throws -> [T]
    func save(entity: T) async throws
    func delete(entity: T) async throws
}

实现仓库接口

我们一般会根据数据的存放方式来定义不同的仓库实现,比如对于网络请求的数据,定义一种仓库的实现,对于本地数据库中存放的数据定义一种仓库的实现,也可以定义一种假数据仓库来编写测试代码。

比如对于网络请求的数据,可以定义一个如下的仓库实现:

class DefaultNewsListRepository: NewsListRepository {
    let remoteDataSource: NewsListRemoteAPI
    init(remoteDataSource: NewsListRemoteDataSource) {
        self.remoteDataSource = remoteDataSource
    }
    func readNewsList() async throws -> [News] {
        return remoteDataSource.requestNewsList();
    }
}

对于本地数据,可以设计一个如下的仓库实现:

class DatabaseNewsListRepository: NewsListRepository {
    let newsListDataStore: NewsListDataStore
    init(newsListRemoteAPI: NewsListRemoteAPI) {
        self.newsListRemoteAPI = newsListRemoteAPI
    }
    func readNewsList() async throws -> [News] {
        return newsListRemoteAPI.requestNewsList();
    }
}

为了编写测试代码,可以提供一个假数据的仓库实现:

class FakeNewsListRepository: NewsListRepository {
    func readNewsList() async throws -> [News] {
        return [
            News(),
            News(),
            News()
        ]
    }
}

如果需要从接口请求到数据后放入本地数据库缓存,然后从本地数据库中读取数据渲染在界面上,也可以用一个仓库搞定。

class DefaultNewsListRepository: NewsListRepository {
    let newsListRemoteAPI: NewsListRemoteAPI
    let newsListDataStore: NewsListDataStore
    init(newsListRemoteAPI: NewsListRemoteAPI, newsListDataStore: NewsListDataStore) {
        self.newsListRemoteAPI = newsListRemoteAPI
        self.newsListDataStore = newsListDataStore
    }
    func readNewsList() async throws -> [News] {
        var newsList = newsListDataStore.readNewsList()
        if newsList.count == 0 {
            let news = newsListRemoteAPI.requestNewsList()
            newsListDataStore.save(newsList)
            newsList = news
        }   
        return newsList
    }
}

选择用哪个仓库实现

在现代 App 项目中,一般会用 MVVM 等架构来组织代码。这里以 MVVM 为例,ViewModel 会依赖仓库接口来存取数据。

class NewsListViewModel: ViewModel {
    let newsListRepository: NewsListRepository
    init(newsListRepository: NewsListRepositoy) {
        self.newsListRepository = newsListRepository
    }
}

为了提高代码的维护性和扩展性,最好使用依赖注入的方式来给 ViewModel 注入 Repository 的依赖,这样可以方便得替换仓库的实现而不用修改 ViewModel 的代码。

可以在创建 ViewModel 时创建对应的仓库对象,也可以使用依赖注入容器。

init(newsListRepository: NewsListRepository = DIContainer.shared.resolve(NewsListRepository.self)) {
    self.newsListRepository = newsListRepository
}

处理数据源的变更

当遇到需要变更数据源的时候,例如本地数据库从 CoreData 切换到 SQLite 或 Realm。或者更换了网络库,从 NSURLSession 换成 Alamofire,这些情况仓库模式就能发挥它的优势。无需修改业务方代码,只需要替换成一种新的仓库实现即可。

还有另一种情况,就是对于同一个业务,后端协议变更了。如果使用了仓库模式,也可以很方便的进行代码调整。

通常,业务层会有一个业务模型,比如对于用户的信息,在业务层定义了一套模型:

struct DomainUser {
    let name: String
    let age: Int
    let nickname: String
}

原本通过接口返回的 json 字符串:

{ 
    "name": "zhangsan",
    "age": 20,
    "nickname": "xiaozhang"
}

可以直接通过解析 JSON 然后构造出 DomainUser 对象,但是突然某一天后端说要技术调整,迁移到新的接口,新接口返回的结构和以前不一样了。

如果没有用仓库模式,业务方直接依赖具体的数据模型,如果接口结构调整了,那么所有的业务调用方的代码都要调整。

使用了仓库模式,业务方依赖于仓库,仓库可以在获取到数据结构后将它转为业务方需要的数据模型,这样无论后端协议怎么变更,都可以仅在数据层增加一种新的仓库实现,不需要改动业务方代码。这遵守了开放-封闭程序设计原则。

总结

在稍具规模的项目中使用仓库模式,可以让代码抽象度更高,耦合度更低,方便扩展和维护,可以编写测试代码,在大型项目中,可以方便的实现数据源和数据结构的切换。仓库模式也将数据存储的细节和程序的其它部分分离开,使得职责更清晰。

仓库模式也有一些缺点:为了实现这一模式需要编写更多的代码,增加了代码复杂性。需要编写映射代码来讲数据映射为业务模型。如果是小型的项目就不需要使用仓库模式了。

参考资料

  • iOS: Repository pattern in Swift
  • Repository Pattern in Swift
  • github.com/kudoleh/iOS…
  • Repository pattern in Swift

以上就是仓库模式及其在Swift 项目中的应用详解的详细内容,更多关于Swift 项目仓库模式的资料请关注编程网其它相关文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

仓库模式及其在Swift项目中的应用详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

仓库模式及其在Swift项目中的应用详解

这篇文章主要为大家介绍了仓库模式及其在Swift项目中的应用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-01-28

Golang Facade模式在中大型项目中的应用场景

在中大型项目中,Golang Facade模式可以应用于以下场景中:1. 复杂系统的简化接口:中大型项目通常包含多个子系统,每个子系统都有自己的接口和实现逻辑。Facade模式可以将这些复杂的子系统封装在一个统一的接口之后,为上层提供一个简
2023-10-08

PHP 防抖技术的实现原理及其在项目中的应用

防抖技术(Debounce)是一种常用于前端开发的技术,其作用是在某个事件被触发后,延迟执行相应的操作,直到事件停止触发一定时间后才真正执行。这种技术常被用于减少频繁触发事件带来的性能影响,提升用户体验。而在 PHP 中同样可以实现防抖技术
2023-10-21

PHP 防抖技术的优势及其在实际项目中的应用

随着移动互联网的快速发展,用户对于web应用程序的要求也越来越高。随着用户在web应用程序中的操作频繁增加,前端错误的增加也是不可避免的。这导致程序的连续,重复的请求,从而占用系统资源,降低程序的性能和响应速度。为了解决这个问题,我将会介绍
2023-10-21

Oracle主目录的功能及其在数据库管理中的应用

Oracle数据库是一款功能强大的关系型数据库管理系统,提供了许多高级功能来管理和操作数据库。其中,Oracle主目录(Oracle Directory)是一个重要的功能模块,用于管理数据库中的文件和目录。本文将介绍Oracle主目录的功能
Oracle主目录的功能及其在数据库管理中的应用
2024-03-07

详解MVP模式在Android开发中的应用

一、MVP介绍 随着UI创建技术的功能日益增强,UI层也履行着越来越多的职责。为了更好地细分视图(View)与模型(Model)的功能,让View专注于处理数据的可视化以及与用户的交互,同时让Model只关系数据的处理,基于MVC概念的MV
2022-06-06

设计模式在大型软件项目中的应用指南

设计模式在大型软件项目中至关重要,可提高代码复用性、可维护性和可扩展性。常用的设计模式包括:单例模式:确保仅存在一个特定类的实例。工厂方法模式:创建一个对象的接口,无需指定其具体类。观察者模式:允许对象订阅事件并接收通知。代理模式:提供替代
设计模式在大型软件项目中的应用指南
2024-05-10

PHP 防抖技术的实现方案及其在项目中的应用效果

随着互联网的快速发展,Web应用程序的用户交互变得越来越频繁。用户在页面上的一些操作可能会触发后台数据库的读写操作,而这些操作的频繁发生可能会对系统造成压力。因此,为了保证系统的稳定性和响应速度,我们需要对用户的操作进行控制,防止频繁触发后
2023-10-21

单例模式在PHP项目中的应用场景与扩展思考

引言单例模式是一种常见的设计模式,它用于限制类的实例化次数,确保在整个应用程序中只有一个实例存在。在PHP项目中,单例模式可以应用于各种场景,如数据库连接、配置文件读取、日志记录等。本文将介绍单例模式在PHP项目中的应用场景,同时探讨如何扩
2023-10-21

C++ 成员函数详解:对象方法在设计模式中的应用

c++++ 成员函数在设计模式中的应用包括:封装数据、避免重复代码和提高可测试性。实战案例中,工厂模式通过成员函数实现:抽象产品接口定义共同行为,具体产品类实现具体行为,工厂根据类型创建产品,客户使用成员函数创建和使用产品。C++ 成员函数
C++ 成员函数详解:对象方法在设计模式中的应用
2024-04-29

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录