【问题标题】:How to instantiate a class with asynchronous Firebase methods in Swift?如何在 Swift 中使用异步 Firebase 方法实例化一个类?
【发布时间】:2019-09-13 07:55:02
【问题描述】:

我正在通过 Firebase DataSnapshot 实例化 User 类。在调用初始化程序init(snapshot: DataSnapshot) 时,它应该通过getFirebasePictureURLgetFirebaseNameString 方法的@escaping 完成处理程序(使用Firebase 的observeSingleEvent 方法)从两个不同的数据库引用中异步检索值,即pictureRefnameRef )。为了避免在所有成员初始化之前被闭包捕获的'self'错误,我不得不使用""URL(string: "initial") 的临时值初始化fullNamepictureURL。但是,当通过User(snapshot: DataSnapshot) 实例化类时,这些值实际上从未使用检索到的 Firebase 值进行更新。

import Firebase

class User {

 var uid: String
 var fullName: String? = ""
 var pictureURL: URL? = URL(string: "initial")

//DataSnapshot Initializer

init(snapshot: DataSnapshot) {

self.uid = snapshot.key

getFirebasePictureURL(userId: uid) { (url) in

    self.getFirebaseNameString(userId: self.uid) { (fullName) in

        self.fullName = fullName
        self.profilePictureURL = url

    }
}

func getFirebasePictureURL(userId: String, completion: @escaping (_ url: URL) -> Void) {

    let currentUserId = userId
    //Firebase database picture reference
    let pictureRef = Database.database().reference(withPath: "pictureChildPath")

    pictureRef.observeSingleEvent(of: .value, with: { snapshot in

        //Picture url string
        let pictureString = snapshot.value as! String

        //Completion handler (escaping)
        completion(URL(string: pictureString)!)

    })

}


func getFirebaseNameString(userId: String, completion: @escaping (_ fullName: String) -> Void) {

    let currentUserId = userId
    //Firebase database name reference
    let nameRef = Database.database().reference(withPath: "nameChildPath")

    nameRef.observeSingleEvent(of: .value, with: { snapshot in

        let fullName = snapshot.value as? String

       //Completion handler (escaping)
        completion(fullName!)

        })
     }
  }

发生这种情况是否有原因,我将如何解决这个问题,以便它初始化为检索到的值,而不是仅仅保留临时值?是不是因为init不是异步的?

编辑:我正在从 Firebase 数据库的一个节点读取数据,并使用该数据创建一个新的节点子节点。初始化 User 类的方法会在数据库中创建这个新节点为:

如您所见,子项使用临时值更新,因此程序执行似乎不等待回调。

任何帮助将不胜感激!

【问题讨论】:

  • 有一个答案可能会有所帮助,但这里似乎有很多不必要的代码。也许需要它,但您是否有一个用户 uid、用户名和个人资料图片 url 存储在三个单独的 firebase 节点中的情况?此外,您有一个类 var self.uid,这很好,一个类 var,不需要将它传递给任何函数,因为它对整个类都可用,而且它是第一件事.你是在课外调用你的函数吗?
  • @Jay 谢谢!我意识到我没有必要使用uid 作为论据。这三个数据点都是单个节点的子节点,因此很多代码可能是多余的。不会从类外部调用 get 方法——仅在类实例化时调用。
  • 如果所有数据点都包含在一个节点中,那么大部分代码都是不必要的。让我尝试一个答案,看看这是否让代码更易于管理。
  • 更新了我的答案,但问题中有一些未知数;为什么 -Lomc6GG..." 有 *author 的子节点?父节点是什么?您正在使用回调,但可能不需要,特别是如果您的代码正在观察包含作者数据,因为如果添加了新作者,它将通过事件传递到您的应用程序。关于发布的另一件事:我们经常需要在答案中使用结构,因此它们应该作为文本包含,这样我们就不必重新键入它们。要获取您的 Firebase 结构,请使用 Firebase 控制台->导出 JSON 并复制并粘贴您的结构的 sn-p。

标签: swift firebase firebase-realtime-database


【解决方案1】:

通过 cmets,我们似乎可以大大减少代码,这也将使其更易于管理

(见编辑)

从一个更简单的 User 类开始。请注意,它是通过传递快照然后读取子节点并填充类变量来初始化的

class UserClass {
    var uid = ""
    var username = ""
    var url = ""

    init?(snapshot: DataSnapshot) {
        self.uid = snapshot.key
        self.username = snapshot.childSnapshot(forPath: "fullName").value as? String ?? "No Name"
        self.url = snapshot.childSnapshot(forPath: "url").value as? String ?? "No Url"
    }
}

然后是从 Firebase 读取用户并创建单个用户的代码

func fetchUser(uidToFetch: String) {
    let usersRef = self.ref.child("users")
    let thisUserRef = usersRef.child(uidToFetch)
    thisUserRef.observeSingleEvent(of: .value, with: { snapshot in
        if snapshot.exists() {
            let user = UserClass(snapshot: snapshot)
            //do something with user...
        } else {
            print("user not found")
        }
    })
}

我不知道用户是如何被使用的,但是如果您需要在 Firebase 闭包之外对用户执行其他操作,您可以添加一个完成处理程序

func fetchUser(uidToFetch: String completion: @escaping (UserClass?) -> Void) {
    //create user
    completion(user)

编辑:

根据其他信息,我将更新答案。从重申目标开始。

OP 有两个节点,一个节点存储用户信息,例如姓名,另一个单独的节点存储图片的 url。他们想从第一个节点获取名称,从第二个节点获取图片 url,并创建一个新的第三个节点,其中包含这两个数据以及 uid。这是图片的可能结构

pictureUrls
   uid_0: "some_url/uid_0"
   uid_1: "some_url/uid_1"

然后我们将使用上面相同的 /users 节点。

这是从 /users 读取名称的代码,来自 /pictureUrls 的图片 url 将它们组合在一起并写出一个带有 /author 子节点的新节点,其中包含该数据和 uid。

func createNode(uidToFetch: String) {
    let usersRef = self.ref.child("users")
    let thisUserRef = usersRef.child(uidToFetch)

    let imageUrlRef = self.ref.child("pictureUrls")
    let thisUsersImageRef = imageUrlRef.child(uidToFetch)

    let allAuthorsRef = self.ref.child("allAuthors")

    thisUserRef.observeSingleEvent(of: .value, with: { snapshot in
        let userName = snapshot.childSnapshot(forPath: "name").value as? String ?? "No Name"

        thisUsersImageRef.observeSingleEvent(of: .value, with: { imageSnap in
            let imageUrl = imageSnap.value as? String ?? "No Image Url"

            let dataToWrite = [
                "full_name": userName,
                "profile_picture": imageUrl,
                "uid": uidToFetch
            ]

            let thisAuthorRef = allAuthorsRef.childByAutoId()
            let authorRef = thisAuthorRef.child("author")
            authorRef.setValue(dataToWrite)
        })
    })
}

firebase 的输出是这样的

allAuthors
   -LooqJlo_Oc-voUHai3k //created with .childByAutoId
      author
         full_name: "Leroy"
         profile_picture: "some_uid/uid_0_pic"
         uid: "uid_0"

与问题中显示的输出完全匹配。

我删除了错误检查以缩短答案,所以请重新添加它,我也省略了回调,因为不清楚为什么需要它。

【讨论】:

  • 非常感谢!但是,我应该澄清一下,传递给 init 方法的数据快照不是包含fullNamepictureURL 的数据快照。它是数据库中不同节点的数据快照。我会更新帖子以澄清。
  • @SergioCharles 根据您的更新,我用与问题匹配的完整答案编辑了问题。
【解决方案2】:

这很hacky。

您应该在init 方法中添加completionHandler。因此,当您的异步调用完成时,您将获得对象的实际值。

init(snapshot: DataSnapshot, completionHandler: @escaping (User) -> Void) {

    self.uid = snapshot.key

    getFirebasePictureURL(userId: uid) { (url) in

        self.getFirebaseNameString(userId: self.uid) { (fullName) in

            self.fullName = fullName
            self.profilePictureURL = url

            completionHandler(self)
        }
    }
}

希望对你有帮助。

【讨论】:

  • 这实际上是解决这个问题的一种非常惯用的方法,所以我不认为是hacky。赞成
  • @SagarChauhan 非常感谢!如果在一个方法内,比如setUp(),我初始化类并从FPUser 获取回调,我是否必须使用调度组以便程序等待方法内的回调?例如。 func setUp() { anotherFunction() ; FPUser { (user) in //do something with the user} }
  • 虽然我真的很喜欢这个答案,但根据来自 OP 的 cmets 和其他信息,它现在与 actual 问题无关。 OP 正在尝试从两个不同的节点提取数据并将它们组合在一起并写出第三个节点,因此此处的完成处理程序将不适用,因为它可能会或我不会在读取另一个节点之前触发。可以更新吗?
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-09-19
  • 1970-01-01
相关资源
最近更新 更多