public 异步加载图像的函数和 returns UIImage
public function that loads image async and returns UIImage
我正在尝试编写 public 加载图像异步和 returns 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)
我得到的不是字符串参数,而是 class?为什么会这样?
你问:
Instead of String argument I get the class? Why is that so?
这是因为你把它当作一个 class 函数来调用,但没有这样定义它。如果在函数声明中添加 class
关键字,您将看不到该行为。
但这里还有更深层次的问题:您不能 return 来自异步函数的值(因为该函数将 return 立即完成,而异步请求稍后才会完成)。
一个解决方案是提供一个完成处理程序,如果成功检索到图像,将调用该处理程序:
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
}
}
注意,上面假设不可能在中间的时间段内在 table 中插入一行。如果该假设无效,而不是使用旧的 indexPath
,您可能会重新查询您的模型以确定在调用异步完成处理程序时哪些 IndexPath
对该单元格有效。
对于 Swift 2 版本,请参阅此答案的 previous revision。
我正在尝试编写 public 加载图像异步和 returns 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)
我得到的不是字符串参数,而是 class?为什么会这样?
你问:
Instead of String argument I get the class? Why is that so?
这是因为你把它当作一个 class 函数来调用,但没有这样定义它。如果在函数声明中添加 class
关键字,您将看不到该行为。
但这里还有更深层次的问题:您不能 return 来自异步函数的值(因为该函数将 return 立即完成,而异步请求稍后才会完成)。
一个解决方案是提供一个完成处理程序,如果成功检索到图像,将调用该处理程序:
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
}
}
注意,上面假设不可能在中间的时间段内在 table 中插入一行。如果该假设无效,而不是使用旧的 indexPath
,您可能会重新查询您的模型以确定在调用异步完成处理程序时哪些 IndexPath
对该单元格有效。
对于 Swift 2 版本,请参阅此答案的 previous revision。