从UITable返回的异步函数中提取数据

时间:2019-06-19 04:36:47

标签: ios swift asynchronous

我有一个名为collectData的函数,该函数在Firebase数据库上的目录和数据中进行迭代。因为它使用异步函数,所以返回数据并不是一个显而易见的选择。我使用以前问题中的其他解决方案来确定我需要一个回调系统。这似乎可行,但是,我需要为UITableView的行数返回数组的长度,这需要返回。这再次导致异步功能出现问题。请查看下面的代码,如果有其他可行的方法或者是否需要澄清,请告诉我。

收集数据的功能:

func gatherData(following: Array<String>, completionHandler: @escaping (_ results: NSArray) -> ()) {
    var returnData: Array<Any> = []
    for follow in following {
        let refr = Database.database().reference(withPath: "/users/\(follow)/")
        refr.observeSingleEvent(of: .value, with: { (snapshot) in
            let values = snapshot.value as! NSDictionary
            let postID = values["currentPostNumber"] as! Int
            var count: Int = 0
            if (postID > 1) {
                count = postID - 1
            }
            for index in 0...count {
                let ref = Database.database().reference(withPath: "/posts/\(follow)/\(index)/")
                ref.observeSingleEvent(of: .value, with: { (snapshot) in
                    let value = snapshot.value as! NSDictionary
                    let captions = value["caption"] as! String
                    let pathToUserImg = value["userImg"] as! String
                    let refre = Storage.storage().reference(withPath: "\(pathToUserImg)")
                    refre.getData(maxSize: 1 * 1024 * 1024) { (data, error) in
                        let image = UIImage(data: data!) as Any
                        let insertIntoArray = [index, captions, image]
                        returnData.append(insertIntoArray)
                        if (index == count) {
                            completionHandler(returnData as NSArray)
                        }
                    }
                })
            }
        })
    }
}

tableView: enter image description here

2 个答案:

答案 0 :(得分:1)

首先声明一个数组来存储API的dataArray

var dataArray = [Any]()

永远不要像这样在UITableView's datasource方法中调用API。让函数说,

func getDataFromAPI() {
   //Call your api function here
   gatherData(following: [""]) { results in 
       self.dataArray = results
       self.yourTableView.reload() //Then reload your tableView
   }
}

最后像这样修改datasource方法的主体:

override func tableView(_ tableView: UITableView,...) -> Int {
   return dataArray.count
}

然后使用dataArray在控制器中填充数据。

答案 1 :(得分:1)

无法在numberOfRowsInSection内部使用补全代码返回值。

您需要先在viewDidLoad中调用API。成功调用API后,您需要重新加载数据。请参考下面的代码。

   gatherData(following: [""]) { results in 
       self.aryResult = results
       self.tableView.reload()
   }

现在,您只需要在self.aryResult.count方法内返回numberOfRowsInSection