我试图在调试区域显示我的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值即可。每次我在调试区域的右侧输出中运行po bbupData
时,它只是部分地显示了我在JSON文件中的20个值中的3个值。当我在位于同一swift文件末尾的completion(jnslst)
中添加另一个断点并按继续执行程序时,出现以下错误: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个值(不知道是否应该显示该方式)。当我继续执行程序到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)
更新2
它似乎只通过jnsextrct
读取一个值并将其返回到jnslst
。然后,当我在下一个屏幕截图中添加另一个断点时,它将转到下一个值。当我在第二个屏幕截图中输入po jnslst
时,它会返回一个[BonBonUp.JSONExtrct]
,而不是之前更新中的20个。
更新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
在我添加nil
方法之前使用func
但我收到以下错误: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!, () -> _)'
答案 0 :(得分:6)
错误很明显,您正在尝试将字符串转换为字典。在json中,您发布了密钥" image"," image2"和" image3"是字符串。
{
"titulo": "Verox's 1002",
"image": "http://wlodsgn.x10host.com/img/jeans/vrxjns1002/veroxjeans1002_front.jpg",
"image2": "http://wlodsgn.x10host.com/img/jeans/vrxjns1002/veroxjeans1002_rightside.jpg",
"image3": "http://wlodsgn.x10host.com/img/jeans/vrxjns1002/veroxjeans1002_front_b.jpg",
"marca": "Verox",
"color": "Azul",
"tipo": "Jean",
"ref": 1002
}
所以获取这些值的正确方法是:
if let let imageURL = data["image"] as? String {
self.imageUrl = imageURL
}
如果您不确定要尝试转换的对象的类型,请避免使用显式解包。
<强>更新强>
新错误与前一个错误具有相同的性质,您隐式地展开了一个nil值,在这种情况下是闭包completion
。你对闭包的定义是completion: ((AnyObject) -> Void)!
对编译器说:
当你需要使用这个闭包时,不要担心它肯定会有非零值。
但是你在这里传递nil
值:
json.loadBbup(nil)
隐式展开很有用,但是你以错误的方式使用它。如果您希望完成闭包可以是nil
,则必须以这种方式将其定义为可选:
completion: ((AnyObject) -> Void)?
并以这种方式调用它:
completion?(jnslst)
关于数组只包含3个值而不是20的事实,有很多原因。我建议在此行jnslst.append(jnsextrct)
中添加一个断点,并观察jnsextrct
的值。
作为一般建议: