Swift-使用完成处理程序更新闭包外的全局变量

Posted

技术标签:

【中文标题】Swift-使用完成处理程序更新闭包外的全局变量【英文标题】:Swift- updating global variable outside closure with completion handler 【发布时间】:2018-03-01 07:11:49 【问题描述】:

我已经搜索过了,但是我无法弄清楚为什么当我调用这个函数时我的全局数组变量没有在全局范围内得到更新。我已经在代码中注释了可以打印()我想要的结果但不能使其在函数之外可用的地方。我知道这很简单,我不了解闭包和完成处理程序,但我需要其他人来看看。谢谢。

import Foundation
import PlaygroundSupport

func pdate(month: String, day: String, year: String) -> String 
    let pdate = month + "/" + day + "/" + year
    return pdate


struct Phenomenon 
    var phenom: String
    var date: String
    var time: String


struct Apsides: Codable 
    let data: [Apsdata]?
    struct Apsdata: Codable 
        let phenom: String
        let year: String
        let month: String
        let day: String
        let time: String
    


var arrFromURL: [(Phenomenon)] = []  //define a global I can use later- I know, globals are evil

func doURLsessions(completionHandler: @escaping (Phenomenon?)->()) 
    let sessionApsides = URLSession(configuration: URLSessionConfiguration.default)
    if let timeurl = URL(string: "http://api.usno.navy.mil/seasons?year=2018&tz=-8&dst=true") 
        (sessionApsides.dataTask(with: timeurl, completionHandler:  (data, response, error) in
            if let error = error 
                print("Error: \(error)")
             else if let data = data 
                let jsonDecoder = JSONDecoder()
                let response = try? jsonDecoder.decode(Apsides.self, from: data)
                if (response?.data != nil) 
                    for item in [response?.data] 
                        for p in item! 
                            switch p.phenom 
                            case "Perihelion":
                                let perihelion = Phenomenon.init(
                                    phenom: p.phenom, date: pdate(month: p.month, day: p.day, year: p.year), time: p.time)
                                completionHandler(perihelion)
                            case "Aphelion":
                                let aphelion = Phenomenon.init(
                                    phenom: p.phenom, date: pdate(month: p.month, day: p.day, year: p.year), time: p.time)
                                completionHandler(aphelion)
                            default: ()
                            
                        
                    
                
            
            //print(arrFromURL) //This works fine, but not what I want
        )).resume()
    


func chPhenom(phenom: String, date: String, time: String) 
    let p = Phenomenon.init(phenom: phenom, date: date, time: time)
    arrFromURL.append(p)
    //print(arrFromURL) //Also works fine,  but I want it's contents avilable outside this function


doURLsessions(completionHandler: Phenomenon in
    chPhenom(phenom: Phenomenon!.phenom, date: Phenomenon!.date, time: Phenomenon!.time)
)

print(arrFromURL) //This just prints []- why isn't the global var available here? Is it printed before the asynch func finishes?

PlaygroundPage.current.needsIndefiniteExecution = true

【问题讨论】:

该 URL 似乎不起作用。甚至来自浏览器。给出超时错误。 我刚试过,“api.usno.navy.mil/seasons?year=2018&tz=-8&dst=true”。似乎对我有用? @ChrisTaylor 你没看错,print(arrFromURL) 在调用completionHandler 之前被调用。所以你只需打印空数组。 【参考方案1】:

函数doURLsessions 中的代码是异步的——它不会阻止其他代码执行,而且由于它是网络请求,因此需要一些时间才能完成。在您的情况下,print(arrFromURL)doURLsessions 完成之前被调用,所以 arrFromURL 仍然是空的。

【讨论】:

谢谢。如何使 doURLsessions 生成的内容在其范围或完成处理程序范围之外可用? 你要么必须等待并在 doURLsessions 完成后访问结果,要么持续监控 arrFromURL 并在其值发生变化时执行某些操作。 但是您的 arrFromURLdoURLsessions 完成处理程序中可用,我会从那里继续您的工作:doURLsessions(completionHandler: Phenomenon in chPhenom(phenom: Phenomenon!.phenom, date: Phenomenon!.date, time: Phenomenon!.time) refresh() ) func refresh() print(arrFromURL) 好点-我需要访问数组的内容来更新我的视图控制器中的标签,我已经建立了 IBAction 连接,我似乎无法从我的完成处理程序中做到这一点关闭。正如你所建议的,我会尝试监控阵列的更新。

以上是关于Swift-使用完成处理程序更新闭包外的全局变量的主要内容,如果未能解决你的问题,请参考以下文章

在 Swift 中遇到完成处理程序和闭包问题

何时在 Swift 中使用全局变量

当完成处理程序显式使用 @escaping 时,Swift 推断完成处理程序闭包是默认的 @nonescaping 而不是 @escaping

Swift-- 闭包

iOS Swift 如何访问在完成处理程序闭包中创建的数据——在闭包之外

Swift 完成处理程序 - 转义尾随闭包