我试图在调试区域显示我的JSON信息,但我遇到了一个问题,我没有找到解决方案 .
首先,让我展示我到目前为止的代码:
BBUpJSON.swift
import Foundation
class BBUpJSON {
func loadBbup(completion: ((AnyObject) -> Void)!) {
var urlString = "http://xxxdsgn.xxxhost.com/json/jnslst.json"
let session = NSURLSession.sharedSession()
let bbupUrl = NSURL(string: urlString)
var task = session.dataTaskWithURL(bbupUrl!){
(data, response, error) -> Void in
if error != nil {
println(error.localizedDescription)
} else {
var error : NSError?
var bbupData = NSJSONSerialization.JSONObjectWithData(data, options: .MutableContainers, error: &error) as! NSArray
var jnslst = [JSONExtrct]()
for jnsextrct in bbupData{
let jnsextrct = JSONExtrct(data: jnsextrct as! NSDictionary)
jnslst.append(jnsextrct)
}
let priority = DISPATCH_QUEUE_PRIORITY_DEFAULT
dispatch_async(dispatch_get_global_queue(priority, 0)) {
dispatch_async(dispatch_get_main_queue()) {
completion(jnslst)
}
}
}
}
task.resume()
}
}
JSONExtrct.swift
import Foundation
class JSONExtrct {
var titulo : String!
var imageUrl : String!
var imageData : NSData?
var imageUrl2 : String!
var imageData2 : NSData?
var imageUrl3 : String!
var imageData3 : NSData?
var marca : String!
var color : String!
var tipo : String!
var ref : Int!
init(data : NSDictionary) {
self.titulo = getStringFromJSON(data, key: "titulo")
let image = data["image"] as! NSDictionary
self.imageUrl = getStringFromJSON(image, key: "image")
let image2 = data["image2"] as! NSDictionary
self.imageUrl2 = getStringFromJSON(image, key: "image2")
let image3 = data["image3"] as! NSDictionary
self.imageUrl3 = getStringFromJSON(image, key: "image3")
self.marca = getStringFromJSON(data, key: "marca")
self.color = getStringFromJSON(data, key: "color")
self.tipo = getStringFromJSON(data, key: "tipo")
self.ref = data["ref"] as! Int
}
func getStringFromJSON(data: NSDictionary, key: String) ->String{
//let info : AnyObject? = data[key]
if let info = data[key] as? String {
return info
}
return ""
}
}
ViewController.swift
import UIKit
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
let json = BBUpJSON()
json.loadBbup(nil)
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
}
因此,当我在 BBUpJSON.swift 文件中使用 for jnsextrct in bbupData{
中的断点运行应用程序时,它会在调试区域的左侧显示我有 20 values . 每次我在调试区域的右侧输出中运行 po bbupData
时,它只是部分显示了我在JSON文件中的20个值中的3个值 . 当我在位于同一swift文件末尾的 completion(jnslst)
中添加另一个断点并按 Continue program execution 时,出现以下错误: Could not cast value of type '__NSCFString' (0x10c3c2c50) to 'NSDictionary' (0x10c3c38d0)
. 然后它将错误重定向到位于我的 JSONExtrct.swift 中的 let image = data["image"] as! NSDictionary
.
这是尝试提取信息的json文件的link .
-----UPDATE------
将 JSONExtrct.swift 中的字符串值修改为:
if let imageUrl = data["image"] as? String {
self.imageUrl = imageUrl
}
if let imageUrl2 = data["image2"] as? String {
self.imageUrl2 = imageUrl2
}
if let imageUrl3 = data["image3"] as? String {
self.imageUrl3 = imageUrl3
}
并添加相同的断点以查看 BBUpJSON.swift 中的结果 . 从断点 for jnsextrct in bbupData{
开始,当我在输出区域输入 po bbupData
时,它仍然显示我所拥有的20个相同的3个值(不知道是否应该显示该方式) . 当我 Continue program execution 到 completion(jnslst)
时,它没有显示我之前的错误,但它在键入 po jnslst
时显示以下输出结果:
(lldb) po jnslst
[BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct, BonBonUp.JSONExtrct]
当我继续并在最后一个括号添加一个断点时,它显示以下错误: fatal error: unexpectedly found nil while unwrapping an Optional value
并将错误重定向到 completion(jnslst)
显示 Thread 1: EXC_BAD_INSTRUCTION (code=EXC_I386_INVOP,subcode=0x0)
UPDATE 2
它似乎只通过 jnsextrct
读取一个值并将其返回到 jnslst
. 然后,当我在下一个屏幕截图中添加另一个断点时,它将转到下一个值 . 当我在第二个屏幕截图中键入 po jnslst
时,它会返回一个 [BonBonUp.JSONExtrct]
而不是之前更新中的20个 .
UPDATE 3
已经将 CollectionView
添加到 Main.StoryBoard
但是当我尝试将JSON信息(还没有图像)上传到每个网格时,我收到以下错误 . 在 JnsGridController.swift 下,为了从我的 JSON
文件接收信息,我必须创建一个回调为 didLoadJns
的新方法 . 以下是该代码的一部分:
jns = [JsonExtrct]()
let json = JnsJson()
json.loadJns(didLoadJns)
}
func didLoadJns(jns: [JsonExtrct]){
self.jns = jns
collectionView.reloadData()
在 json.loadJns(didLoadJns)
中, didLoadJns
在添加 func
方法之前使用 nil
但是我收到以下错误: Cannot invoke 'loadJns' with an argument list of type '(([JsonExtrct]) -> ())'
同样在 JnsJson.swift 中,因为我在 JnsGridController 中添加了回调函数,它将从 JsonExtrct
而不是 AnyObject
接收数组,所以我将函数方法从 func loadJns(completion: ((AnyObject) -> Void)!) {
更改为 func loadJns(completion: ((JsonExtrct) -> Void)!) {
,现在我得到了与 JnsGridController 中的一个类似的错误: Cannot invoke 'dispatch_async' with an argument list of type '(dispatch_queue_t!, () -> _)'
1 回答
错误很明显,您正在尝试将字符串转换为字典 . 在您发布了密钥“image”的json中,“image2”和“image3”是字符串 .
所以获取这些值的正确方法是:
如果您不确定要尝试转换的对象的类型,请避免使用显式展开 .
UPDATE
新错误与前一个错误具有相同的性质,您隐式地展开了一个nil值,在这种情况下是闭包
completion
. 您对闭包的定义是completion: ((AnyObject) -> Void)!
对编译器说:但是你在这里传递一个
nil
值:json.loadBbup(nil)
隐式展开很有用,但是你以错误的方式使用它 . 如果您希望完成闭包可以是
nil
,则必须以这种方式将其定义为可选:completion: ((AnyObject) -> Void)?
并以这种方式调用它:
completion?(jnslst)
关于数组只包含3个值而不是20个的事实,有很多原因 . 我建议在这一行
jnslst.append(jnsextrct)
中加一个断点,并观察jnsextrct
的值 .作为一般建议:
当你不需要修改变量时,