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
并在其值发生变化时执行某些操作。
但是您的 arrFromURL
在 doURLsessions
完成处理程序中可用,我会从那里继续您的工作:doURLsessions(completionHandler: Phenomenon in chPhenom(phenom: Phenomenon!.phenom, date: Phenomenon!.date, time: Phenomenon!.time) refresh() ) func refresh() print(arrFromURL)
好点-我需要访问数组的内容来更新我的视图控制器中的标签,我已经建立了 IBAction 连接,我似乎无法从我的完成处理程序中做到这一点关闭。正如你所建议的,我会尝试监控阵列的更新。以上是关于Swift-使用完成处理程序更新闭包外的全局变量的主要内容,如果未能解决你的问题,请参考以下文章
当完成处理程序显式使用 @escaping 时,Swift 推断完成处理程序闭包是默认的 @nonescaping 而不是 @escaping