为什么我的iOS应用程序无法从Firebase数据库读取?

时间:2018-11-24 23:33:36

标签: ios swift xcode firebase

我目前正在麻烦阅读并将Firebase数据库显示在表列表中。下面是我在当前View Controller中使用的代码。

import UIKit
import Firebase
import FirebaseDatabase

class Buy_1: UIViewController, UITableViewDataSource, 
 UITableViewDelegate   {

var ref: DatabaseReference!
var databaseHandle: DatabaseHandle!

var postItem: [String] = []

@IBOutlet weak var TableView: UITableView!

@IBAction func goBackToOneButtonTapped(_ sender: Any) {
    performSegue(withIdentifier: "unwindSegueToVC1", sender: self)
}

override func viewDidLoad() {
    super.viewDidLoad()

    if FirebaseApp.app() == nil {
        FirebaseApp.configure()
    }

    //Set Self as tableview's data source and delegate
    self.TableView.delegate = self
    self.TableView.dataSource = self

    //Set the firebase reference
    ref = Database.database().reference()

    //Retrieve and listen for changes
    databaseHandle = ref?.child ("items").observe(.childAdded, with: { (snapshot) in

        //Convert value of data into string
        if let item = snapshot.value as? String
        {
            self.postItem.append(item)
            self.TableView.reloadData()
            self.ref?.keepSynced(true)
        }

        /*if let actualItem = item {
            //Appends data to our Item Array
            self.postItem.append(actualItem)

            //Reload TableView
            self.TableView.reloadData()
        }*/

    })
    // Do any additional setup after loading the view.
}


override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}

/*
// MARK: - Navigation

// In a storyboard-based application, you will often want to do a little preparation before navigation
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    // Get the new view controller using segue.destinationViewController.
    // Pass the selected object to the new view controller.
}
*/

//MARK: - UITableView Delegate Methods


func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return postItem.count
}

func tableView(_ TableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = TableView.dequeueReusableCell(withIdentifier: "BasicCell", for: indexPath)

    cell.textLabel?.text = postItem[indexPath.row]
    return cell

  }
}

screenshot of Firebase Database

我没有在代码中收到任何错误,所以我不确定是什么错误。 &我已经将数据库规则设置为true。

谢谢。

2 个答案:

答案 0 :(得分:0)

您正在观察.childAdded,它仅在将子级添加到观察到的数据树中时才触发块。 将其更改为.value

答案 1 :(得分:0)

代码中的问题是项的子项不是字符串。它们本身就是childSnapshots或key:value对。

下面是一些示例代码,用于从与您类似的结构中仅读取itemName

items
   item_0
      itemName: "Josh"
   item_1
      itemName: "Screaming Eagle"
   item_2
      itemName: "Insignia"

以及用于打印每个子节点的键和itemName的代码

let ref = self.ref.child("items")
ref.observe(.childAdded, with: { snapshot in
    if snapshot.exists() {
        let dict = snapshot.value as! [String: Any]
        let itemName = dict["itemName"] as? String ?? "No Item Name"
        print(snapshot.key, itemName)
    }
})

和输出

item_0 Josh
item_1 Screaming Eagle
item_2 Insignia

请注意,如果未找到itemName,我将使用nil合并运算符(??)来填充它。

我将快照的值部分视为字典,以允许您访问子节点,但是您也可以使用.childSnapshot这样实现

let i = snapshot.childSnapshot(forPath: "itemName").value as? String ?? "No Item"

在此特定示例中,snapshot.exists是多余的,但应与按.value读取一起使用。