Xcode Swift:无法将类型'__NSCFString'(0x102e8ac50)的值转换为'NSDictionary'(0x102e8b8d0)

时间:2015-08-03 21:40:10

标签: arrays json xcode swift nsdictionary

我试图在调试区域显示我的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

Breakpoint Line 1

Breakpoint Line 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!, () -> _)'

1 个答案:

答案 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的值。

作为一般建议:

  • 避免使用隐式展开,除非您确定该变量为非零值。
  • 当您不需要修改变量时,请使用let代替var;通过这种方式,您可以确定价值永远不会改变。