3

我的 swift 程序一直有这个错误,我使用 PHP 和 MySQL 作为数据库。我要将数据从数据库显示到表格视图。它以前可以工作,但是在将它运行到模拟器几次之后,我遇到了同样的错误

class Home: UIViewController, UITableViewDelegate, UITableViewDataSource {

@IBOutlet weak var btnaddclass: UIButton!
@IBOutlet var tableView: UITableView!
   var values: NSArray = []

func get(){
    let url = NSURL(string: "http://localhost/show_db.php")
    let data = NSData(contentsOf: url! as URL)
    values = try! JSONSerialization.jsonObject(with: data! as Data, options:JSONSerialization.ReadingOptions.mutableContainers)as! NSArray
    tableView.reloadData()
}
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return values.count
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath) as! SpecialCell
    let maindata = values[indexPath.row] as! [String:AnyObject]

    cell.Lblclasstitle.text = maindata["class_title"] as? String
    cell.Lblschool.text = maindata["school"] as? String
    cell.Lblsubject.text = maindata["subject"] as? String
    return cell
}
   override func viewDidLoad() {
    self.view.addGestureRecognizer(self.revealViewController().panGestureRecognizer())
    get()


}
override func viewWillAppear(_ animated: Bool) {
    self.tableView.reloadData()
}
}
4

3 回答 3

8

正如其他一些答案所指出的那样,问题是您正在强制 JSON 反序列化并假设它总是 100% 的时间工作。这几乎总是不是一个好的选择。有两种选择可以解决这个问题:

1)使用尝试?反而

如果您尝试执行的语句失败,这可能是一个不错的选择,那么您将获得 nil,然后可以根据它做一些事情。例如,您可以将 get() 函数更改为:

func get(){
    //remove NS prefix and unwrap optional
    guard let url = URL(string: "http://localhost/show_db.php") else { return }

    //remove ! from url and unwrap optional
    guard let data = try? Data(contentsOf: url) else { return }

    //try to deserialize. The return value is of type Any
    guard let deserializedValues = try? JSONSerialization.jsonObject(with: data) else { return }

    //Convert to array of dictionaries
    guard let arrayOfDictionaryValues = deserializedValues as? [[String:String]] else { return }

    //If you make it here, that means everything is ok
    values = arrayOfDictionaryValues
    
    tableView.reloadData()
}

请注意,每个守卫语句都为您提供了执行某些操作并返回的机会。也许您想显示一条消息,说明存在网络错误。

2) 使用 do try catch 块。

我认为使用尝试?更适合您的情况,但在某些情况下,do try catch 块可能是合适的。例如:

func getWithErrorHandling() {
    guard let url = URL(string: "http://localhost/show_db.php") else { return }

    do {
        let data = try Data(contentsOf: url)
        let deserializedValues = try JSONSerialization.jsonObject(with: data)
        guard let arrayOfDictionaryValues = deserializedValues as? [[String:String]] else { return }
        values = arrayOfDictionaryValues
    } catch {
        //You should separate the different catch blocks to handle the different types of errors that occur
        print("There was an error")
    }
}

可以对您的整体方法进行其他改进,但由于您是初学者(如您所述),最好一次学习一件事。

此外,正如其他人所提到的,您不应该从主线程获取数据。请改用 URLSession。这篇文章就是一个很好的例子:

在 Swift 3 中正确解析 JSON

于 2017-01-15T14:19:09.537 回答
5

NS...1.不要在 Swift3 中使用类。例如:

let url = URL(string: "http://localhost/show_db.php")
let data = Data(contentsOf: url!)

2.使用do-try-catch 处理异常。

do {
    try ...
} catch {
    // handle exception
}

as!3.除非你确定它是哪个类,否则不要使用。用这个:

if let array = some as? Array {
    // use array now
}
于 2017-01-15T13:08:53.547 回答
1

您的代码充满了问题。

“试一试!” 如果您正在调用的代码引发异常,则表达式会崩溃。这就是你崩溃的原因。不要那样做,尤其是在处理来自外部来源的数据时。改为使用do { try.. } catch { }。在 Apple 的文档和在线文档中有大量的 swift try/catch 块示例。

您收到的错误消息告诉您出了什么问题。您在 JSON 数据流的末尾收到垃圾字符。将您的数据流转换为字符串并记录下来。

不要使用as!任何一种,除非您确定可以将对象转换为所需的类型。(强制施法失败时会崩溃。)

正如@OOPer 在他们的评论中所说,您不应该从主线程上的 URL 加载数据。这会锁定 UI 直到加载完成,如果加载时间过长,可能会导致您的应用程序被终止。您应该使用异步方法,例如URLSession.

于 2017-01-15T13:24:24.747 回答