【问题标题】:Discrepancy passing data between view controllers视图控制器之间传递数据的差异
【发布时间】:2016-03-02 08:51:27
【问题描述】:

我正在尝试在我的第一个 ViewController 和我的 SecondViewController 之间传递数据。我在 ViewController 的顶部有两个变量,它们被初始化为空。然后我获取解析 JSON 数据并将这两个变量的值设置为解析的临时 JSON 变量。然后在 prepareforSegue 函数中,让 SecondViewController 的变量等于我的第一个 ViewController 中的前两个变量。由于某种原因,传递的变量仅在变量为空时传递。我不确定我是否有意义,但这是我的代码:

class ViewController: UIViewController {

var stockSymbol = String()
var stockPrice = String()

@IBOutlet weak var textField: UITextField!

@IBAction func buttonPressed(sender: AnyObject) {

    let requestURL: NSURL = NSURL(string: "https://finance.yahoo.com/webservice/v1/symbols/\(textField.text!)/quote?format=json")!
    let urlRequest: NSMutableURLRequest = NSMutableURLRequest(URL: requestURL)
    let session = NSURLSession.sharedSession()
    let task = session.dataTaskWithRequest(urlRequest) {
        (data, response, error) -> Void in

        let httpResponse = response as! NSHTTPURLResponse
        let statusCode = httpResponse.statusCode

        if (statusCode == 200) {
            print("Everyone is fine, file downloaded successfully.")

            do {

                let json = try NSJSONSerialization.JSONObjectWithData(data!, options:.AllowFragments)

                if let resources = json["list"]?!["resources"] as? [[String:AnyObject]] {
                    if let fields = resources[0]["resource"]?["fields"] as? [String:String], price = fields["price"], symbol = fields["symbol"] {

                        self.stockSymbol = symbol
                        self.stockPrice = price

                        print(self.stockSymbol, self.stockPrice)
                    } 
                }

            } catch {
                print("Error with Json: \(error)")

                //end catch
            }

        } //end if status code == 200
    } //end task


    task.resume()

}

override func viewDidLoad() {
    super.viewDidLoad()

    //var test:String

}

override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject!) {
    if (segue.identifier == "searchResults") {

        print("working \(self.stockPrice)")
        let secondVC = segue.destinationViewController as! SecondViewController;
        secondVC.passedSymbol = self.stockSymbol
        secondVC.passedPrice = self.stockPrice
        print(secondVC.passedSymbol)



    }
}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}


}

基本上我希望将股票价格和代码转移到我的第二个视图控制器的变量中,但它似乎不起作用。

第二个视图控制器的代码:

class SecondViewController: UIViewController {

var passedSymbol: String!
var passedPrice: String!

@IBOutlet weak var symbol: UILabel!

@IBOutlet weak var price: UILabel!

override func viewDidLoad() {
    super.viewDidLoad()

    print("passed \(passedSymbol)")
    print("second view")
    dispatch_async(dispatch_get_main_queue()) {
    self.symbol.text = self.passedSymbol
    self.price.text = self.passedPrice

    }

}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}


}

当我将“aapl”放入 searchField 时的输出语句

working 

passed 
second view
Everyone is fine, file downloaded successfully.
AAPL 100.529999

【问题讨论】:

  • 请显示SecondViewController 的代码以及任何打印语句的输出。

标签: ios json swift


【解决方案1】:

首先,不要这样做:

var stockSymbol = String()
var stockPrice = String()

这是一个坏习惯,它会阻止编译器抱怨未初始化的变量,但完全忽略了 Swift 中选项的强大功能。

相反,这样做:

var stockSymbol:String!
var stockPrice:String!

这将确保您在尝试使用变量而不设置值时遇到异常。

现在,您得到空值的原因(或者如果您进行了我的建议更改会出现异常)是您在按下按钮时启动异步数据获取但立即启动 segue,大概是因为您有与按钮以及touchUpInside 处理程序关联的操作。

您需要删除动作转场并通过 ctrl 将场景顶部的黄色 View Controller 图标拖动到目标视图控制器来创建程序转场,并像往常一样为新转场提供标识符。

然后,您可以更改 buttonPressed 函数以在检索到数据后调用 segue;

@IBAction func buttonPressed(sender: AnyObject) {

    let requestURL: NSURL = NSURL(string: "https://finance.yahoo.com/webservice/v1/symbols/\(textField.text!)/quote?format=json")!
    let urlRequest: NSMutableURLRequest = NSMutableURLRequest(URL: requestURL)
    let session = NSURLSession.sharedSession()
    let task = session.dataTaskWithRequest(urlRequest) {
        (data, response, error) -> Void in

        let httpResponse = response as! NSHTTPURLResponse
        let statusCode = httpResponse.statusCode

        if (statusCode == 200) {
            print("Everyone is fine, file downloaded successfully.")

            do {

                let json = try NSJSONSerialization.JSONObjectWithData(data!, options:.AllowFragments)

                if let resources = json["list"]?!["resources"] as? [[String:AnyObject]] {
                    if let fields = resources[0]["resource"]?["fields"] as? [String:String], price = fields["price"], symbol = fields["symbol"] {

                        self.stockSymbol = symbol
                        self.stockPrice = price

                        print(self.stockSymbol, self.stockPrice)
                        dispatch_async(dispatch_get_main_queue(), {
                            self.performSegueWithIdentifier("searchResults",sender:sender)
                        })
                    } 
                }

            } catch {
                print("Error with Json: \(error)")

                //end catch
            }

        } //end if status code == 200
    } //end task


    task.resume()

}

【讨论】:

    【解决方案2】:

    原因应该是你去第二个viewcontroller的时候,还没有拿到第一个viewcontroller里面的数据。 dataTaskWithRequest 是一个异步函数,这意味着它在另一个线程而不是主 ui 线程上运行。你可以做的是等到你在第一个 viewcontroller 获得数据,然后调用 segue。控制台上的打印信息可能会让你感到困惑,好像你已经得到了数据,但实际上它可能是在调用 segue 之后发生的,导致没有数据传递。

    【讨论】:

    • 什么意思删除异步?它本身是异步的。
    猜你喜欢
    • 2011-07-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-02-12
    相关资源
    最近更新 更多