【问题标题】:Parsing JSON Swift解析 JSON 斯威夫特
【发布时间】:2014-10-24 15:19:51
【问题描述】:

我正在开发一个显示实时比特币价格的应用程序。我正在使用 2 个 API 来执行此操作 - 一个纯文本和一个 JSON。我在使用 JSON API 时遇到了一些问题。

这是我的一些 Swift 代码

func BTCFallback(){

    var string2 = currencySelector.currentTitle


    var url = NSURL(string:"https://bitpay.com/api/rates/" +  (string2)!)
    var request = NSURLRequest(URL: url)


    NSURLConnection.sendAsynchronousRequest(request, queue: NSOperationQueue(), completionHandler:fallback)

    var data = NSData(contentsOfURL:url);
    let value = NSString(string: USD.text).doubleValue / NSString(data:data, encoding:NSUTF8StringEncoding).doubleValue

    // Define JSON string
    var JSONString = "\(data)"

    // Get NSData using string
    if let JSONData = JSONString.dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: false) {

        // Parse JSONData into JSON object
        var parsingError: NSError?
    if let JSONObject = NSJSONSerialization.JSONObjectWithData(JSONData, options: nil, error: &parsingError) as? [String: AnyObject] {

        // If the parsing was successful grab the rate object
        var rateObject: AnyObject? = JSONObject["rate"]

        // Make sure the rate object is the expected type
        if let rate = rateObject as? Float {
            println("rate is \(rate)")
            BTC.text = "\(rate)"
        }
    } else {

        // There was an error parsing the JSON data
        println("Error parsing JSON: \(parsingError)")
        BTC.text = "err1"
    }

}






}

在上述代码中,currencySelector.currentTitle 等于 ISO 货币代码,例如 USD。 BTC.text 是一个 UI 元素。

预期的行为是代码会将“rate”的对应项设置为 BTC.text 的文本。如果这有帮助,API 会返回类似 {"code":"USD","name":"US Dollar","rate":376.71} 的内容。使用上面的示例,我希望将 BTC.text 设置为 376.71

这是发生了什么:consoe 给出了错误 Error parsing JSON: Optional(Error Domain=NSCocoaErrorDomain Code=3840 "The operation couldn’t be completed. (Cocoa error 3840.)" (JSON text did not start with array or object and option to allow fragments not set.) UserInfo=0x16eb0f60 {NSDebugDescription=JSON text did not start with array or object and option to allow fragments not set.})

我做错了什么?提前致谢!

【问题讨论】:

  • 数据在传入解析器之前是什么样子的?
  • 1.为什么要将数据转换为字符串,然后再转换为JSONObjectWithData 之前的数据? 2. 打印 JSON 字符串 JSONString 并检查 JSON 并验证它是有效的 JSON。
  • @Hyperbole JSON 看起来像 API 中的 {"code":"USD","name":"US Dollar","rate":376.71}。 API 的链接是bitpay.com/api/rates/USD
  • @rocket101 不能这样看,否则解析器不会出错。确保您实际上传递了一个包含数据的数据对象。在dataUsingEncoding() 之后放置一个断点并在那里检查。
  • @rocket101 是的,这就是编码的字符串。您是否有理由不将contentsOfURL 调用中的NSData 对象直接传递给JSONObjectWithData()?翻译中可能会丢失一些东西。

标签: ios json xcode parsing swift


【解决方案1】:

这都是处理返回数据和反序列化的问题。

这里是示例代码,注意Optionals的处理应该更好,这只是为了演示基本代码。例如,我使用了一个简单的同步网络调用。:

var url: NSURL! = NSURL(string:"https://bitpay.com/api/rates/AUD")
var request = NSURLRequest(URL: url)
var response: NSURLResponse?
var error: NSError?
var data: NSData? = NSURLConnection.sendSynchronousRequest(request, returningResponse:&response, error:&error)
println("data: \(data)")

if let data: NSData = NSURLConnection.sendSynchronousRequest(request, returningResponse:&response, error:&error) {
    println("data: \(data)")

    var parsingError: NSError?
    if let rateDictionary = NSJSONSerialization.JSONObjectWithData(data, options: nil, error: &parsingError) as NSDictionary? {
        println("rateDictionary: \(rateDictionary)")

        // If the parsing was successful grab the rate object
        if var rateString: AnyObject = rateDictionary["rate"] {
            println("rateString: \(rateString)")

            // Make sure the rate object is the expected type
            if let rate = rateString.floatValue {
                println("rate is \(rate)")
            }
        }
    }
}

输出:

数据:可选(7b22636f 6465223a 22415544 222c226e 616d6522 3a224175 73747261 6c69616e 20446f6c 6c617222 2c227261 7465223a 34330372e 3939) 率字典:{ 代码 = 澳元; name = "澳元"; 率=“407.9917”; } 速率字符串:407.9917 率为 407.992

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2023-03-07
    • 2017-02-15
    • 2017-07-22
    • 2017-03-24
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多