将数据解析为tableview swift 3

时间:2017-01-24 21:17:28

标签: json uitableview swift3 xcode8-beta2

我试图解析网站上的数据,然后按下按钮将其显示在桌面视图中。我使用swift 3,Xcode 8.2 beta并且无法将数据存储到数组中或显示到tableView中。这是我的tableViewCell类:

class TableViewCell: UITableViewCell {
@IBOutlet weak var userIdLabel: UILabel!
@IBOutlet weak var titleLabel: UILabel!
override func awakeFromNib() {
    super.awakeFromNib()
    // Initialization code
}

这是我的viewController代码:

import UIKit
class SecondViewController: UIViewController, UITableViewDelegate,UITableViewDataSource {
let urlString = "https://jsonplaceholder.typicode.com/albums"
@IBOutlet weak var tableView: UITableView!
  var titleArray = [String]()
  var userIdArray = [String]()
@IBAction func getDataButton(_ sender: Any) {
    self.downloadJSONTask()
     self.tableView.reloadData()
}
override func viewDidLoad() {
    super.viewDidLoad()
     tableView.dataSource = self
     tableView.delegate = self
}
override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
}

func downloadJSONTask() {
    let url = NSURL(string: urlString)
    var downloadTask = URLRequest(url: (url as? URL)!, cachePolicy:  URLRequest.CachePolicy.reloadIgnoringCacheData, timeoutInterval: 20)
    downloadTask.httpMethod = "GET"


    URLSession.shared.dataTask(with: (url! as URL),  completionHandler: {(Data, URLResponse, Error) -> Void in
        let jsonData = try? JSONSerialization.jsonObject(with: Data!,  options: .allowFragments)
           print(jsonData as Any)
        if let albumArray = (jsonData! as AnyObject).value(forKey: "") as? NSArray {
            for title in albumArray{
                if let titleDict = title as? NSDictionary {
                    if let title = titleDict.value(forKey: "title") {
                        self.titleArray.append(title as! String)
                        print("title")
                        print(title)
                    }
                    if let title = titleDict.value(forKey: "userId")    {
                        self.userIdArray.append(title as! String)
                    }
                    OperationQueue.main.addOperation ({
                        self.tableView.reloadData()
                    })
                }
            }                
        }        
    }).resume()       
    }
 func tableView(_ tableView: UITableView, numberOfRowsInSection  section: Int) -> Int{
    return titleArray.count
  }
  func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "cell") as! TableViewCell
    cell.titleLabel.text = titleArray[indexPath.row]
    cell.userIdLabel.text = userIdArray[indexPath.row]
    return cell
    }
    }

1 个答案:

答案 0 :(得分:0)

您的代码中存在许多问题,最糟糕的是在Swift中使用NSArray/NSDictionary

JSON是一个字典数组,键title的值是String userID的值是Int,所以你必须声明你的数组

var titleArray = [String]()
var userIdArray = [Int]()

永远不要将JSON数据投射到大多数未指定的Any,这是另一个不行。将它始终投射到实际类型。另一个大问题是闭包中的Data参数与Swift3中的本机结构冲突。使用始终小写参数标签。您的代码中根本不使用该请求。在Swift 3中,始终使用原始结构URLDataURLRequest等等。最后.allowFragments是无意义的,因为JSON清楚地以集合类型开始。

let url = URL(string: urlString)!
let request = URLRequest(url: url, cachePolicy: .reloadIgnoringCacheData, timeoutInterval: 20)
URLSession.shared.dataTask(with: request) { (data, response, error) in
    if error != nil {
        print(error!)
        return
    }

    do {
        if let jsonData = try JSONSerialization.jsonObject(with:data!, options: []) as? [[String:Any]] {
            print(jsonData)
            for item in jsonData {

                if let title = item["title"] as? String {
                    titleArray.append(title)
                }
                if let userID = item["userId"] as? Int {
                    userIdArray.append(userID)
                }
                DispatchQueue.main.async {
                    self.tableView.reloadData()
                }
            }
        }
    } catch let error as NSError {
        print(error)
    }
}.resume()

PS:使用两个独立的数组作为数据源也很可怕。想象一下,其中一个可选绑定可能会失败,并且数组中的项数将不同。这是一个非常有趣的运行时崩溃邀请。