iOS swift, как я могу дождаться асинхронной задачи внутри функции, которая нуждается в возвращаемом значении
Я использую Swift 3.0 и создал функцию, которая возвращает массив целых чисел. Массивы целых чисел очень специфичны, и они получены из базы данных, поэтому HTTP-вызов является асинхронным. Это функция, потому что я использую ее в 3 разных контроллерах, поэтому имеет смысл написать ее один раз. Моя проблема заключается в том, что асинхронный код возвращается после оператора возврата внизу, поэтому он возвращает ноль. Я попробовал приведенный здесь пример. Ожидание завершения задачи, однако она не работает, главным образом потому, что мне нужно вернуть значение. Это мой код
func ColorSwitch(label: [UILabel]) -> [Int] {
for (index, _) in label.enumerated() {
label[index].isHidden = true
}
// I need the value of this variable in the return
// statement after the async is done
var placeArea_id = [Int]()
let urll:URL = URL(string:ConnectionString+"url")!
let sessionn = URLSession.shared
var requestt = URLRequest(url: urll)
requestt.httpMethod = "POST"
let group = DispatchGroup()
group.enter()
let parameterr = "http parameters"
requestt.httpBody = parameterr.data(using: String.Encoding.utf8)
let task = sessionn.dataTask(with:requestt, completionHandler: {(data, response, error) in
if error != nil {
print("check check error")
} else {
do {
let parsedData = try JSONSerialization.jsonObject(with: data!, options: .allowFragments) as? [String:Any]
DispatchQueue.main.async {
if let Profiles = parsedData?["Results"] as? [AnyObject] {
if placeArea_id.count >= 0 {
placeArea_id = [Int]()
}
for Profiles in Profiles {
if let pictureS = Profiles["id"] as? Int {
placeArea_id.append(pictureS)
}
}
}
group.leave()
}
} catch let error as NSError {
print(error)
}
}
})
task.resume()
group.notify(queue: .main) {
// This is getting the value however can't return it here since it
// expects type Void
print(placeArea_id)
}
// this is nil
return placeArea_id
}
Я уже проверил, и значения возвращаются внутри асинхронного кода, теперь просто нужно вернуть его, любые предложения будут хорошими.
2 ответа
Вы захотите использовать для этого замыкания или изменить свою функцию на синхронную.
func ColorSwitch(label: [UILabel], completion:@escaping ([Int])->Void) {
completion([1,2,3,4]) // when you want to return
}
ColorSwitch(label: [UILabel()]) { (output) in
// output is the array of ints
print("output: \(output)")
}
Вот довольно хороший блог о закрытии http://goshdarnclosuresyntax.com/
Вы не можете заставить вашу функцию возвращать значение из асинхронной операции внутри этой функции. Это победило бы цель асинхронности. Для того, чтобы передать эти данные за пределы вашего ColorSwitch(label:)
функция, вам также нужно, чтобы она принимала замыкание, которое будет вызываться по завершении, которое принимает [Int]
в качестве параметра. Объявление вашего метода должно выглядеть примерно так:
func ColorSwitch(label: [UILabel], completion: @escaping ([Int]) -> Void) -> Void {
for (index, _) in label.enumerated() {
label[index].isHidden = true
}
var placeArea_id = [Int]()
let urll:URL = URL(string:ConnectionString+"url")!
let sessionn = URLSession.shared
var requestt = URLRequest(url: urll)
requestt.httpMethod = "POST"
let group = DispatchGroup()
group.enter()
let parameterr = "http parameters"
requestt.httpBody = parameterr.data(using: String.Encoding.utf8)
let task = sessionn.dataTask(with:requestt, completionHandler: {(data, response, error) in
if error != nil {
print("check check error")
} else {
do {
let parsedData = try JSONSerialization.jsonObject(with: data!, options: .allowFragments) as? [String:Any]
DispatchQueue.main.async {
if let Profiles = parsedData?["Results"] as? [AnyObject] {
if placeArea_id.count >= 0 {
placeArea_id = [Int]()
}
for Profiles in Profiles {
if let pictureS = Profiles["id"] as? Int {
placeArea_id.append(pictureS)
}
}
}
group.leave()
completion(placeArea_id) // This is effectively your "return"
}
} catch let error as NSError {
print(error)
}
}
})
task.resume()
}
Позже вы можете назвать это так:
ColorSwitch(label: []) { (ids: [Int]) in
print(ids)
}