Как установить результат в виде массива в BFTask of Bolts iOS Framework?
У меня есть метод, в котором я хочу вернуть массив структур при успешном завершении обратного вызова Alamofire.
func getPopularMedias() -> BFTask {
let instaUrl: String = "https://api.instagram.com/v1/media/popular"
let user: PFUser = currentUser()
let accessToken = user.objectForKey("accessToken") as! String
var medias: [Media] = []
let task = BFTaskCompletionSource()
Alamofire.request(.GET, instaUrl, parameters: ["access_token": accessToken])
.responseJSON { request, response, data in
var json = JSON(data.value!)
for(_, subJson): (String, JSON) in json["data"] {
let image: UIImage = NSURL(string: subJson["images"]["low_resolution"]["url"].stringValue)
.flatMap { NSData(contentsOfURL: $0) }
.flatMap { UIImage(data: $0) }!
let profileImage: UIImage = NSURL(string: subJson["user"]["profile_picture"].stringValue)
.flatMap { NSData(contentsOfURL: $0) }
.flatMap { UIImage(data: $0) }!
medias.append(Media(name: subJson["user"]["full_name"].stringValue, image: image, profileImage: profileImage))
}
task.setResult(medias)
}
return task.task
}
При настройке task.setResult
Я получаю сообщение об ошибке cannot convert value of [Media] to expected argument AnyObject!
В настоящее время я использую XCode 7.0 GM и Swift 2.
2 ответа
Проблема здесь в том, что task.setResult
принимает AnyObject!
, который требует классов, или массив классов. Структуры не попадают в эту категорию (на самом деле это будет работать с Any
целеуказатель). Вы можете конвертировать ваши Media
структурировать класс или инкапсулировать его в класс.
Вот одна вещь, которую я бы предложил вам сделать в этом конкретном случае. Я вижу, что вы, похоже, используете SwiftyJSON, поэтому вы могли бы передать json.rawData() в результате выполнения задачи и впоследствии обработать ее. Вот пример из жизни. Я получаю некоторые данные из своей базы данных, получаю другой набор данных, основанный на первоначально полученных данных, и отправляю их обратно в основную функцию как json.rawData(). Afterwards, I am able to process my new data once I convert my NSData back to JSON.
ps You could also utilize a completionHandler, that will allow you to return whatever you want.
func startLoadingItunesDataFor(postsQuery: PFQuery, completionHandler: ((posts: [Post]) -> Void)) {
var newPosts = [Post]()
postsQuery.findObjectsInBackground().continueWithSuccessBlock({ (task: BFTask!) -> AnyObject! in
var tasks = [BFTask]()
if let result = task.result {
let posts = result as! [PFObject]
for post in posts {
let tempPost = Post(
//////
)
newPosts.append(tempPost)
tasks.append(self.getMovieInfoByITunesID(post["trackID"] as! Int))
}
}
return BFTask(forCompletionOfAllTasksWithResults: tasks)
}).continueWithSuccessBlock({ (task: BFTask!) -> AnyObject! in
let results = task.result as! NSArray
for (index, postData) in results.enumerate() {
let json = JSON(data: postData as! NSData)
////
}
completionHandler(posts: newPosts)
return nil
})
}
func getMovieInfoByITunesID(iTunesID: Int) -> BFTask {
let mainTask = BFTaskCompletionSource()
ITunesApi.lookup(iTunesID).request({ (responseString: String?, error: NSError?) -> Void in
if
// error == nil,
let responseString = responseString, let dataFromString = responseString.dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: false) {
let json = JSON(data: dataFromString)
do {
try mainTask.setResult(json["results"][0].rawData())
}
catch {
}
} else {
// process error
}
})
return mainTask.task
}