swift - 在 Cocoa Touch 框架中使用 Realm 时的模式

标签 swift cocoa-touch design-patterns realm cocoapods

这是在 Cocoa Touch Framework 中使用惊人的 Realm 数据库时的代码模式/设计问题。具体来说,一个将作为 cocoapod 分发的框架。

假设我在构建的框架中有一些 Realm 对象

public class Dog: Object {
    @objc public private(set) dynamic var name: String?
    public let age = RealmOptional<Int>()
    public private(set) var owner: Person?
}

public class Person: Object {
    @objc public private(set) dynamic var name: String?
    public private(set) var dogs = LinkingObjects(fromType: Dog.self, property: "owner")
}

现在,我希望我的框架的使用者能够与这些对象进行交互。我不想用 MVVM 模式抽象这些,因为我希望我的框架的用户能够利用一些很棒的 Realm 东西,比如查询、排序,最重要的是,Realm 更改通知。

所以,第一个问题。我应该让我的框架的用户直接初始化对象吗?他们将在 Realm 初始值设定项中拥有这些选项,如果他们选择使用它们,他们将对它们负责。但是,我喜欢使用静态方法来使用工厂模式。像这样:

extension Dog {
    public static func retreiveManagedDog() throws -> Dog {
        let dog = Dog()
        do{
            let realm = try Realm()
            realm.beginWrite()

            realm.add(dog)
            try realm.commitWrite()
        }catch{
            throw error
        }
        return dog
    }
}

对于这个用例,这是一个好的设计模式吗?

其次,下一个问题是更新对象。由于所有 Realm 对象都必须在写入事务中更新,我不希望我的框架的用户必须编写一堆样板代码才能更改名称。所以,我写了这样的函数:

//MARK: Extension that has functions to update properties
extension Dog {
    public func updateName(_ name: String?) throws {
        do{
            let realm = try Realm()
            realm.beginWrite()
            self.name = name
            try realm.commitWrite()
        }catch{
            throw error
        }
    }
}

请注意我的对象定义有 private(set) 正是出于这个原因。这将有助于强制我的框架的用户使用我的 setter 方法。

总的来说,我是不是疯了才尝试以这种方式包装 Realm?其他持久存在的伟大框架通常包装所有 SQL Lite/Core 数据逻辑。我还想提出总体上改进此模式的建议。

最佳答案

幸运的是,我认为您的问题没有任何硬性规定的正确答案。这实际上取决于您要做什么。

首先,最好不要篡改默认的 Realm 或默认的 Realm 配置,因为任何直接使用 Realm 的应用程序都可能会操纵其中任何一个。您还需要确保为您的模型类型命名空间并将它们从默认架构中排除。 (同样,您可能希望仅使用您在内部定义的模型架构来打开框架的 Realm 。)

至于您的 API,它的外观将取决于您希望在您的框架上下文中使用 Realm 的目的。

如果 Realm 是一个实现细节,那么控制用户如何通过 API 与您的基于 Realm 的对象交互是有意义的,就像您在上面描述的那样。通过这种方式,您可以准确控制用户可以对模型对象执行的操作,并确保不会以导致其状态变为无效的方式进行修改(如果这是一个问题)。

例如,您可能希望通过您提供的 API 为用户处理在 Realm 对象上注册和取消注册观察者 block ,该 API 指定了自己的先决条件和要求。如果一个对象应该被观察的时间段应该与框架中的某个其他对象或系统一致或受其控制,这可能很有用:而不是告诉用户如何存储他们的通知 token 以及何时是适当的时间处理它们,您可以在幕后处理所有这些细节,并确保用户不会在您的框架上下文中滥用 Realm。

您甚至可以将其发挥到极致,通过使您的 Realm 类型符合某些协议(protocol)并让您的 API 使用和提供该协议(protocol)类型,从而使 Realm 的使用对您的用户不可见。这样,您的用户就无法判断您的对象是 Realm 对象,也不会想将它们粘贴到自己的 Realms 中、复制它们等。

但是,如果您想让用户知道您在 API 中为他们提供了 Realm 对象,甚至可能在他们自己使用 Realm 的同时使用它们(例如,将它们放在他们自己的 Realms 中),那么它明确地将它们作为 Realm 对象出售是有意义的,也许使用像您出于方便原因而严格描述的那样的帮助 API。在这种情况下,您需要考虑框架对 Realm 的使用如何避免干扰消费者对 Realm 的使用。

关于swift - 在 Cocoa Touch 框架中使用 Realm 时的模式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47190808/

相关文章:

swift - Swift 中的 Interface Builder、@IBOutlet 和委托(delegate)和数据源协议(protocol)

iphone - 如果用户点击屏幕键盘,我如何关闭键盘?

通过可变模板继承 C++11 构造函数

ios - Xcode 11.4。导航的标题颜色从 Storyboard中变为黑色

ios - 类型 'Any'没有下标成员NSIndexPath和indexPath Swift 3

ios - 在状态之间向 UIButton 添加动画

design-patterns - 存储库模式是否与 Asp.net 提供程序模型相同?

java - 如何在Java/Android模块化库中使两个相关的包相互独立?

swift - 有没有办法从 Swift 3 中的异步闭包中抛出错误?

iphone - 如何通过餐厅的menuId 获取餐厅?