我正在尝试编写加载图像异步并返回UIImage的公共函数。我想用它来填充UITableView图像。
class CommonFunctions {
func loadImageAsync(StringURL: NSString) -> UIImage{
var returningImage = UIImage()
let url = NSURL(string: StringURL)
let requestedURL = NSURLRequest(URL: url!)
NSURLConnection.sendAsynchronousRequest(requestedURL, queue: NSOperationQueue.mainQueue(), completionHandler: {
response, data, error in
if error != nil {
println("there is some error loading image")
}
else {
if let image = UIImage(data: data){
returningImage = image
}
}
})
return returningImage
}
}
问题是当我想使用这个功能时:
cell.imageModelImage.image = CommonFunctions.loadImageAsync(CommonFunctions)
而不是String参数我得到了这个类?为什么会这样?
答案 0 :(得分:2)
你问:
而不是String参数我得到了这个类?为什么会这样?
这是因为你把它称为类功能,但没有定义它。如果您在函数声明中添加class
关键字,则无法看到该行为。
但是这里存在更深层次的问题:您无法从异步函数返回值(因为该函数将立即返回,而异步请求将在稍后完成)。
一种解决方案是提供一个完成处理程序,如果成功检索到图像,将会调用该处理程序:
class func loadImageAsync(stringURL: String, completion: @escaping (UIImage?, Error?) -> Void) {
let url = URL(string: stringURL)!
URLSession.shared.dataTask(with: url) { data, response, error in
guard let data = data, error == nil else {
DispatchQueue.main.async { completion(nil, error) }
return
}
let image = UIImage(data: data)
DispatchQueue.main.async { completion(image, nil) }
}.resume()
}
注意,我将完成处理程序分派回主队列,在那里应该进行所有UI更新。
然后,在tableView(_:cellForRowAt:)
中,您可以提供将在异步请求完成时调用的块:
cell.imageModelImage.image = ... // remember to initialize image view before starting async method
CommonFunctions.loadImageAsync(with: "http://example.com/test.jpg") { image, error in
if let cell = tableView.cellForRow(at: indexPath) as? CustomCell { // make sure cell hasn't scrolled out of view
cell.imageModelImage.image = image
}
}
注意,上述假设在中间时间内不可能在表格中插入一行。如果该假设无效,而不是使用旧的indexPath
,那么在调用异步完成处理程序时,您可能需要重新查询模型以确定IndexPath
对此单元格有效。
对于Swift 2的演绎,请参阅此答案的previous revision。