Как установить результат в виде массива в 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
}
Другие вопросы по тегам