swift – 异步回调中的Inout参数无法按预期工作

前端之家收集整理的这篇文章主要介绍了swift – 异步回调中的Inout参数无法按预期工作前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我正在尝试使用inout参数插入函数,以将从异步回调接收的数据附加到外部数组.但是,它不起作用.我尝试了所知的一切,找出原因 – 没有运气.

正如@AirspeedVelocity建议的那样,我重写了如下代码删除不必要的依赖项.我还使用Int作为inout参数来保持简单.输出始终是:
c之前:0
c之后:1

我无法弄清楚这里出了什么问题.

  1. func getUsers() {
  2. let u = ["bane","LiweiZ","rdtsc","ssivark","sparkzilla","Wogef"]
  3. var a = UserData()
  4. a.userIds = u
  5. a.dataProcessor()
  6. }
  7.  
  8. struct UserData {
  9. var userIds = [String]()
  10. var counter = 0
  11. mutating func dataProcessor() -> () {
  12. println("counter: \(counter)")
  13. for uId in userIds {
  14. getOneUserApiData(uriBase + "user/" + uId + ".json",&counter)
  15. }
  16. }
  17. }
  18.  
  19. func getOneUserApiData(path: String,inout c: Int) {
  20. var req = NSURLRequest(URL: NSURL(string: path)!)
  21. var config = NSURLSessionConfiguration.ephemeralSessionConfiguration()
  22. var session = NSURLSession(configuration: config)
  23. var task = session.dataTaskWithRequest(req) {
  24. (data: NSData!,res: NSURLResponse!,err: NSError!) in
  25. println("c before: \(c)")
  26. c++
  27. println("c after: \(c)")
  28. println("thread on: \(NSThread.currentThread())")
  29. }
  30.  
  31. task.resume()
  32. }

谢谢.

不幸的是,在async-callback中修改inout参数是没有意义的.

the official document开始:

Parameters can provide default values to simplify function calls and can be passed as in-out parameters,which modify a passed variable once the function has completed its execution@H_404_20@.

An in-out parameter has a value that is passed in to the function,is modified by the function,and is passed back@H_404_20@ out of the function to replace the original value.

在语义上,输入输出参数不是“call-by-reference”,而是“call-by-copy-restore”.

在您的情况下,只有当getOneUserApiData()返回时,计数器才会被写入后备,而不是在dataTaskWithRequest()回调中.

以下是您的代码中发生的情况

>在getOneUserApiData()调用时,将计数器0的值复制到c1
>闭包捕获c1
>调用dataTaskWithRequest()
> getOneUserApiData返回,并且 – unmodified – c1的值被写回到计数器
>对c2,c3,c4重复1-4程序……
> …从互联网上取货……
>调用回调并且c1递增.
>调用回调,c2递增.
>调用回调并且c3递增.
>调用回调并且c4递增.
> ……

因此,计数器未经修改:(

详细解释

通常,in-out参数通过引用传递,但它只是编译器优化的结果.当闭包捕获inout参数时,“传递引用”并不安全,因为编译器无法保证原始值的生命周期.例如,请考虑以下代码

  1. func foo() -> () -> Void {
  2. var i = 0
  3. return bar(&i)
  4. }
  5.  
  6. func bar(inout x:Int) -> () -> Void {
  7. return {
  8. x++
  9. return
  10. }
  11. }
  12.  
  13. let closure = foo()
  14. closure()

在此代码中,当foo()返回时,将释放var i.如果x是对i的引用,则x会导致访问冲突.为了防止这种竞争条件,Swift采用了“逐个复制恢复”策略.

猜你在找的Swift相关文章