我正在使用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
答案 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)
}