首页 文章

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

提问于
浏览
4

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

Breakpoint Line 1

Breakpoint Line 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 回答

  • 6

    错误很明显,您正在尝试将字符串转换为字典 . 在您发布了密钥“image”的json中,“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
    }
    

    如果您不确定要尝试转换的对象的类型,请避免使用显式展开 .

    UPDATE

    新错误与前一个错误具有相同的性质,您隐式地展开了一个nil值,在这种情况下是闭包 completion . 您对闭包的定义是 completion: ((AnyObject) -> Void)! 对编译器说:

    当您需要使用此闭包时,请不要担心它肯定会具有非零值 .

    但是你在这里传递一个 nil 值:

    json.loadBbup(nil)

    隐式展开很有用,但是你以错误的方式使用它 . 如果您希望完成闭包可以是 nil ,则必须以这种方式将其定义为可选:

    completion: ((AnyObject) -> Void)?

    并以这种方式调用它:

    completion?(jnslst)

    关于数组只包含3个值而不是20个的事实,有很多原因 . 我建议在这一行 jnslst.append(jnsextrct) 中加一个断点,并观察 jnsextrct 的值 .

    作为一般建议:

    • 避免使用隐式展开,除非您确定变量为非零值 .
      当你不需要修改变量时,
    • 使用let而不是var;通过这种方式,您可以确定 Value 永远不会改变 .

相关问题