0

我尝试在URLSession.dataTask运行设置 segue 时获取 json 数据。(每个 json 数据作为发送者)

所以首先,我制作了自己的 Class Array productList = [Product]()。接下来,我调用getJsonData()并在其中设置URLSession.dataTask方法。所以我得到了 Parsed json 数据。但是,当我尝试从 保存该 json 数据(将每个数据附加到productList)时dataTask completionHandler,它无法正确保存。(结果productList[]

我想通过 segue 传递解析的 json 数据。我怎样才能做到这一点?

编辑——

class MainVC: UITableViewController {

    var productList = [Product]()

    override func viewDidLoad() {
        super.viewDidLoad()

        getJsonData()

    }

    override func numberOfSections(in tableView: UITableView) -> Int {
        return 1
    }
    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return productList.count
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        if let cell = tableView.dequeueReusableCell(withIdentifier: "ItemCell", for: indexPath) as? ItemCell {
            let product = productList[indexPath.row]

            cell.configureCell(product)

            return cell
        } else {
            return UITableViewCell()
        }
    }


    func getJsonData() {
        let url = URL(string: "http://demo7367352.mockable.io")
        let request = URLRequest(url: url!)
        let defaultSession = URLSession(configuration: URLSessionConfiguration.default)


        let task = defaultSession.dataTask(with: request, completionHandler: { (data, response, error) in

            do {
                guard let data = data, error == nil else {
                    print("network request failed: error = \(error)")
                    return
                }

                guard let rawItem = try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any] else {
                    print("error trying to convert data to JSON")
                    return
                }


                if let fineItem = rawItem["goods"] as? [[String:Any]] {

                    for item in fineItem {
                        let eachProduct = Product(title: "", price: 0)

                        let title = item["TITLE"]
                        let price = item["PRICE"]
                        let regDate = item["REGDATE"]
                        let description = item["DESCRIPTION"]
                        let iconURL = item["ICON_URL"]
                        let images = item["IMAGES"]


                        if let title = title as? String {
                            eachProduct.title = title
                        }
                        if let price = price as? String {
                            eachProduct.price = Int(price)!
                        }

                        DispatchQueue.main.async(execute: {
                            self.productList.append(eachProduct)
                            self.tableView.reloadData()
                        })
                    }

                }

            } catch  {
                print("error trying to convert data to JSON")
                return
            }

        })
        task.resume()
    }

    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        if segue.identifier == "goToProductDetail" {
            if let controller = segue.destination as? DetailVC, let indexPath = tableView.indexPathForSelectedRow {

            }
        }
    }
 }

现在,我可以解析来自URLSession DataTask. 我想实现 tableView 的 segue 来显示细节。但是productList是空的。所以我不能使用prepareForSeguewith productList[indexPath.row]

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "goToProductDetail" {
        if let controller = segue.destination as? DetailVC, let indexPath = tableView.indexPathForSelectedRow {
            controller.product = productList[indexPath.row] // productList is nil.
        }
    }
}
4

3 回答 3

0

您没有发布所有代码,但我相信您的错误是您正在执行异步任务,然后立即在正在修改的数组上调用 print 。在任务完成之前,我不希望数组被填充。

您的 tableView 是否实际填充了结果?您是否打印出 JSON 以确保您的数据正确匹配?是否正在打印错误?

编辑:要沿着 segue 传递您的数据,您需要检索您destinationViewController的变量并将信息传递给它。有一种方法prepareForSegue可以让您在操作发生之前处理初步状态。

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
   let vc = segue.destination as! ExampleVC
   vc.setProducts(productList)
}

类似的东西。显然改变你的类和变量名

于 2017-03-26T14:29:51.447 回答
0

您需要在prepare(for:sender:)那里实现并传递数据:

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if let controller = segue.destination as? MySecondViewController, indexPath = tableView.indexPathForSelectedRow {
        controller.product = productList[indexPath.row]
    }
}

确切的语法会有所不同(目标视图控制器的类名是什么),您必须product在目标中声明该属性,并且目标viewDidLoad需要使用该属性,但希望它说明了基本思想。


一些额外的观察:

  1. 我建议您检查rawItem并确保它是一个字典,其中包含一个被调用的键goods,并且与该键关联的值实际上是一个字典数组。如果没有看到您的 JSON,就不可能说出究竟出了什么问题。

    另外,请考虑:

    if let fineItem = rawItem["goods"] as? [[String:Any]] {
        ...
    }
    

    如果失败了,你永远不会知道。我可能会建议:

    guard let fineItem = rawItem["goods"] as? [[String:Any]] else {
        print("goods not found or wrong type")
        return
    }
    ...
    
  2. productList顺便说一句,与您手头的问题无关,直接在数据任务的完成处理程序中进行变异有点危险。不要从一个线程异步改变从另一个线程读取的数组。数组不是线程安全的。数据任务完成处理程序应该构建一个本地数组,并且只有在完成后,在您将重新加载分派到主队列的地方,您应该在productList重新加载表之前插入代码以替换为您的本地数组。

  3. 此外,您当前正在reloadData解析循环内调用。您通常会在解析循环结束时调用它。现在,如果您的数据集有 100 行,那么您将重新加载表 100 次。

  4. 参考data!有点危险。如果您没有互联网连接,data将是nil并且您的代码将崩溃。我建议:

    guard let data = data, error == nil else {
        print("network request failed: error = \(error)")
        return
    }
    

    然后您可以将data!引用替换为data.

于 2017-03-26T14:53:47.630 回答
0

我解决了我的问题,这是我的最终代码。

class MainVC: UITableViewController {

    var productList = [Product]()

    override func viewDidLoad() {
        super.viewDidLoad()

        getJsonData()

    }

    override func numberOfSections(in tableView: UITableView) -> Int {
        return 1
    }
    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return productList.count
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        if let cell = tableView.dequeueReusableCell(withIdentifier: "ItemCell", for: indexPath) as? ItemCell {
            let product = productList[indexPath.row]

            cell.configureCell(product)

            return cell
        } else {
            return UITableViewCell()
        }
    }

    // parsing
    func getJsonData() {
        let url = URL(string: "http://demo7367352.mockable.io")
        let request = URLRequest(url: url!)
        let defaultSession = URLSession(configuration: URLSessionConfiguration.default)


        let task = defaultSession.dataTask(with: request, completionHandler: { (data, response, error) in

            do {
                guard let data = data, error == nil else {
                    print("network request failed: error = \(error)")
                    return
                }
                guard let rawItem = try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any] else {
                    print("error trying to convert data to JSON")
                    return
                }
                guard let fineItem = rawItem["goods"] as? [[String:Any]] else {
                    print("goods not found or wrong type")
                    return
                }

                for item in fineItem {
                    let eachProduct = Product(title: "", price: 0)

                    let title = item["TITLE"]
                    let price = item["PRICE"]


                    if let title = title as? String {
                        eachProduct.title = title
                    }
                    if let price = price as? String {
                        eachProduct.price = Int(price)!
                    }

                    self.productList.append(eachProduct)
                }

                DispatchQueue.main.async(execute: {
                    self.tableView.reloadData()
                })

            } catch  {
                print("error trying to convert data to JSON")
                return
            }

        })
        task.resume()
    }

    // segue
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        if segue.identifier == "goToProductDetail" {
            if let controller = segue.destination as? DetailVC, let indexPath = tableView.indexPathForSelectedRow {
                controller.product = productList[indexPath.row]
            }
        }
    }
}
于 2017-03-26T17:04:31.987 回答