函数在异步请求完成之前返回

时间:2015-08-04 09:29:10

标签: ios swift asynchronous grand-central-dispatch

我正在使用Twitter API来检索推文列表。该方法应该返回推文列表,但问题是它在请求结束之前返回,这导致返回一个空列表。我该如何解决?这是该方法的源代码:

// Call to search through twitter with a query
func searchQuery(query: String) -> [Tweet] {
    var tweets: [Tweet] = []
    var query_URL = query.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLQueryAllowedCharacterSet())
    if let query_URL = query_URL {
        TwitterClient.sharedInstance.GET("https://api.twitter.com/1.1/search/tweets.json?q=\(query_URL)", parameters: nil, success: { (operation: AFHTTPRequestOperation!, response: AnyObject!) -> Void in
            tweets = parseJSON(response)
            println(tweets.count)
            }, failure: { (operation: AFHTTPRequestOperation!, error: NSError!) -> Void in

        })
    }
    println("ret: \(tweets.count)")
    return tweets
}

在上面的代码中,输出将是

ret: 0
15

我尝试过使用调度组,但我无法让他们工作。这就是我用GCD做的事情:

// Call to search through twitter with a query
func searchQuery(query: String) -> [Tweet] {
    var tweets: [Tweet] = []
    var query_URL = query.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLQueryAllowedCharacterSet())
    var group = dispatch_group_create()
    if let query_URL = query_URL {
        dispatch_group_enter(group)
        TwitterClient.sharedInstance.GET("https://api.twitter.com/1.1/search/tweets.json?q=\(query_URL)", parameters: nil, success: { (operation: AFHTTPRequestOperation!, response: AnyObject!) -> Void in
            tweets = parseJSON(response)
            dispatch_group_leave(group)
            }, failure: { (operation: AFHTTPRequestOperation!, error: NSError!) -> Void in
                dispatch_group_leave(group)
        })
    }

    dispatch_group_wait(group, DISPATCH_TIME_FOREVER)
    return tweets
}

但是,该代码似乎永远等待。

作为参考,Tweet是用于保存与推文相关的数据的结构。我用它以更紧凑的方式移动大量数据。 parseJSON根据JSON响应填充并返回一组推文。理想情况下,返回的数组将保存到tweets,然后应该从方法返回,但这不会发生。

非常感谢任何克服这一点的想法或技巧!

修改:@Hamza Ansari 这是实际的功能:

// Call to search through twitter with a query
func searchQuery(query: String, completionHandler:(returntweets: [Tweet]) -> Void) {
    var query_URL = query.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLQueryAllowedCharacterSet())
    if let query_URL = query_URL {
        TwitterClient.sharedInstance.GET("https://api.twitter.com/1.1/search/tweets.json?q=\(query_URL)", parameters: nil, success: { (operation: AFHTTPRequestOperation!, response: AnyObject!) -> Void in
            var tweets:[Tweet] =  parseJSON(response)
            println("size in method: \(tweets.count)")
            completionHandler(returntweets: tweets)

            }, failure: { (operation: AFHTTPRequestOperation!, error: NSError!) -> Void in

        })
    }
}

但是,当用于初始化数据源的其他方法调用时:

// Initializes the data source
func initialize(query: String) {
    self.query = query
    searchQuery(query, { (returnTweets) -> Void in
        self.searches = returnTweets
        })

    println("size when called: \(searches.count)")
}

输出(按顺序):

size when called: 0
size in method: 15

2 个答案:

答案 0 :(得分:2)

这是设计上的;这是一个"异步"方法(即,它关闭并完成一些工作,并在完成后调用您提供的"成功"代码)。它设计的原因是你的主线程赢了被阻止等待网络,这可能需要不确定的时间。

这需要你相应地设计它周围的代码 - 你不能拥有一个做你想要的方法(你称之为,它返回结果就绪)。好吧,你可以(通过在后台线程上排队异步调用,然后等待它),但是你会等待你不想做的网络。相反,您应该更改自己的方法searchQuery以执行类似操作(异步),并且其调用方应根据需要处理该方法。

当需要完成异步工作时,这种模式既可接受也不好,而且一旦习惯了,它就不会像最初看起来那样繁重。

(参见@ HamzaAnsari的答案,了解如何做到这一点。)

答案 1 :(得分:0)

使用完成处理程序定义函数:

func searchQuery(query: String, completionHandler:(returntweets: [Tweet]) -> Void) {
    var query_URL = query.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLQueryAllowedCharacterSet())
    if let query_URL = query_URL {
      TwitterClient.sharedInstance.GET("https://api.twitter.com/1.1/search/tweets.json?q=\(query_URL)", parameters: nil, success: { (operation: AFHTTPRequestOperation!, response: AnyObject!) -> Void in
        var tweets:[Tweet] =  parseJSON(response)
        println(tweets.count)
        completionHandler(returntweets: tweets)

        }, failure: { (operation: AFHTTPRequestOperation!, error: NSError!) -> Void in

      })
    }
  }

现在打电话给你需要推文的地方:

searchQuery(query: "youQuery",{ (returntweets) -> Void in
      //Do something 
     println(returntweets.count)
    }