我正在开发一个应用程序,它使用Alamofire从几个不同的API中获取数据(每个调用都是使用函数完成的).然后我必须收集所有结果(在我的情况下为Double类型)到一个数组来计算平均值. 只要A
只要Alamofire使用异步调用,就不可能简单地从调用内部向数组添加新值.
这是一个函数,它调用负责通过Alamofire获取数据的每个函数:
func collectData() {
fetchFromFirstAPI()
fetchFromSecondAPI() //etc.
}
以下是其中一个功能的示例:
func fetchFromFirstAPI() {
let APIKey = "XXXXXXXXX"
let APIURL = "http://urlapi.com/api" as URLConvertible
let parameters: Parameters = ["APPKEY": APIKey]
Alamofire.request(APIURL, method: .get, parameters: parameters, encoding: URLEncoding.default).validate().responseJSON { response in
switch response.result {
case .success(let data):
let json = JSON(data)
if let result = json["main"]["value"].double {
myArray.append(result)
} else {
print("error")
}
case .failure:
print("error")
}
}
}
而阵列:
var myArray: [Double] = []
怎么处理呢?
你说:As long as Alamofire uses asynchronous calls, it is impossible to simply append new value to an array from inside of call.
您实际上可以继续将项目附加到完成处理程序内的数组中.并且由于Alamofire默认在主队列上调用其完成处理程序,因此无需进一步同步.您可以使用调度组来了解所有请求何时完成.首先,我会给我的方法完成处理程序,所以我知道它们什么时候完成,例如:
func fetchFromFirstAPI(completionHandler: @escaping (Double?, Error?) -> Void) {
let APIKey = "XXXXXXXXX"
let APIURL = "http://urlapi.com/api"
let parameters: Parameters = ["APPKEY": APIKey]
Alamofire.request(APIURL, parameters: parameters).validate().responseJSON { response in
switch response.result {
case .success(let data):
let json = JSON(data)
if let result = json["main"]["value"].double {
completionHandler(result, nil)
} else {
completionHandler(nil, FetchError.valueNotFound)
}
case .failure(let error):
completionHandler(nil, error)
}
}
}
哪里
enum FetchError: Error {
case valueNotFound
}
然后你可以做类似的事情:
func performRequestsAndAverageResults(completionHandler: @escaping (Double?) -> ()) {
var values = [Double]()
let group = DispatchGroup()
group.enter()
fetchFromFirstAPI { value, error in
defer { group.leave() }
if let value = value { values.append(value) }
}
group.enter()
fetchFromSecondAPI { value, error in
defer { group.leave() }
if let value = value { values.append(value) }
}
group.notify(queue: .main) {
completionHandler(self.average(values))
}
}
func average<T: FloatingPoint>(_ values: [T]) -> T? {
guard values.count > 0 else { return nil }
let sum = values.reduce(0) { $0 + $1 }
return sum / T(values.count)
}
现在,您可能希望以不同方式处理错误,但这应该说明基本思路:只需将结果附加到完成处理程序中,然后使用调度组知道所有请求何时完成,并在通知闭包中执行任何计算想.
显然,上面的代码并不关心这些异步任务的完成顺序.如果您确实关心,则需要稍微调整一下(例如将结果保存到字典中,然后构建排序数组).但是如果你只是对结果求平均值,那么顺序并不重要,而且上面的内容相当简单.
