在TableView上快速显示Firebase数据未显示

时间:2020-06-19 20:15:35

标签: ios swift firebase tableview

我用Firebase作为数据库制作应用程序,我似乎无法将数据显示到tableview中。我检查了我的Firebase,即使我添加了新数据,我的所有数据也都很好,数据立即显示在我的Firebase中。但是似乎我的逻辑有些失落了。...有人可以帮助我吗?

*编辑1行代码无效

这是我的主控制器:

class MainController: UITableViewController, AddPatientControllerr {


private var patientLists = [PatientList]() 
var Segue : String = "PatientName"
var Segue2 : String = "PatientNotes"
let user : User = Auth.auth().currentUser! 

private var rootRef : DatabaseReference!// 1. buat nyambung ke root db

override func viewDidLoad() {
    super.viewDidLoad()
   
    self.rootRef = Database.database().reference() 
    
    populateList()
    
    tableView.delegate = self
    tableView.dataSource = self

}

// MARK : Firebase Function

private func populateList() {
    
    self.rootRef.child(self.user.emailWithoutSpecialChar).observe(.value) { (snapshot) in 
        
        self.patientLists.removeAll()
        
        let pasienListDict = snapshot.value as? [String:Any]  ?? [:] 
        
        for (key,_) in pasienListDict {
            
            if let pasienlistdict = pasienListDict[key] as? [String:Any]{

                if let pasienlist = PatientList(pasienlistdict) { // this line of code is not working 
                    self.patientLists.append(pasienlist)
                   
                }else {
                   print("your condition not working")
                }
            }
            
        }
            DispatchQueue.main.async {
           self.tableView.reloadData()
       }
        
        
    }
    
}

// MARK : Func delegate

func addPatientData(controller: UIViewController, nama: String, tglLahir: String, Telp: String, berat: String, Tinggi: String, golDarah: String) {
       
    let patientList = PatientList(name: nama, tglLahir: tglLahir, Telp: Telp, berat: berat, Tinggi: Tinggi, golDarah: golDarah)
       self.patientLists.append(patientList)
              
       let userRef = self.rootRef.child(self.user.emailWithoutSpecialChar)
       let patientListRef = userRef.child(patientList.name)
       patientListRef.setValue(patientList.toDictionary())
      
       controller.dismiss(animated: true, completion: nil)

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

// MARK : Segue
   
   override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
       
       if segue.identifier == Segue {
           let nc = segue.destination as! UINavigationController
           let addPatientName = nc.viewControllers.first as! ProfileController
           addPatientName.delegate = self
           
           }
           
           else if segue.identifier == Segue2 {
           
           guard let indexPath = self.tableView.indexPathForSelectedRow else {return}
            
           let nc = segue.destination as! PasienProfileController
           nc.pasien = self.patientLists[indexPath.row]
     
       }
   
   }
 

//MARK : TableView

override func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCell.EditingStyle, forRowAt indexPath: IndexPath) {
    if editingStyle == .delete {
        let pasienList = self.patientLists[indexPath.row]
        let pasienListRef = self.rootRef.child(pasienList.name)
        pasienListRef.removeValue()
    }
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    
    return self.patientLists.count
}


override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    
    guard let cell = tableView.dequeueReusableCell(withIdentifier: "Cell") as? MainCell else {return UITableViewCell()}
    let patientListt = self.patientLists[indexPath.row]
    
    cell.NameLbl.text = patientListt.name

    return cell
  }  
}

这是我保留数据和字典的模型:

import Foundation

typealias JSONDictionary  = [String:Any]

class PatientList { 

var name : String!
var tglLahir : String!
var Telp : String!
var berat : String!
var Tinggi : String!
var golDarah : String!
var patientNote :[PatientNote] = [PatientNote]() //ini buat nyimpen notes2 dari tiap2 pasien 

init(name : String, tglLahir : String, Telp : String, berat : String, Tinggi : String, golDarah : String) {
    self.name = name
    self.tglLahir = tglLahir
    self.berat = berat
    self.Tinggi = Tinggi
    self.golDarah = golDarah
    self.Telp = Telp
}

init?(_ dictionary :[String:Any]){
    
    guard let name = dictionary["Name"] as? String else {
        return nil
    }
    guard let berat = dictionary["BeratBadan"] as? String else {
        return nil
    }
    guard let tglLahir = dictionary["TanggalLahir"] as? String else {
        return nil
    }
    guard let Tinggi = dictionary["TinggiBadan"] as? String else {
        return nil
    }
    guard let golDarah = dictionary["GolonganDarah"] as? String else {
        return nil
    }
    guard let Telp = dictionary["Telefon"] as? String else {
        return nil
    }
    self.name = name
    self.berat = berat
    self.tglLahir = tglLahir
    self.Tinggi = Tinggi
    self.golDarah = golDarah
    self.Telp = Telp
    let pasienListDictionary = dictionary["patientNote"] as? [JSONDictionary]
    
    if let dictionaries = pasienListDictionary {
        self.patientNote = dictionaries.compactMap(PatientNote.init)
    }
}

func toDictionary() -> [String:Any] { // ini buat dictionary buat convery object jd string:any, jd biar ga ngubah satu2 kl ada yg salah gituu
    
    return ["Name":self.name, "BeratBadan":self.berat, "TanggalLahir":self.tglLahir, "TinggiBadan":self.Tinggi, "golDarah":self.golDarah, "Telefon":self.Telp, "patientNote":self.patientNote.map{ patientNote in
        return patientNote.toDictionary()
    }]
}

}

这是我的火力基地:

[![Firebase] [1]] [1]

对于初学者来说,我只需要在表格视图中显示我的名字,在这种情况下,我什至无法在表格视图中显示名字数据

我似乎无法找出为什么数据不会显示在表格视图中.... xcode没有显示错误

有人可以帮助我吗?谢谢

firebase Json

"afipermanalivecom" : {
    "Apiyyy" : {
        "BeratBadan" : "",
        "Name" : "Apiyyy",
        "TanggalLahir" : "20-05-2020",
        "Telefon" : "",
        "TinggiBadan" : "",
        "golDarah" : "A+"
    },
    "CocaCola" : {
        "BeratBadan" : "80",
        "Name" : "CocaCola",
        "TanggalLahir" : "20-06-2020",
        "Telefon" : "0878099996049",
        "TinggiBadan" : "190",
        "golDarah" : "A-"
    },
    "Jamsey" : {
        "BeratBadan" : "",
        "Name" : "Jamsey",
        "TanggalLahir" : "19-06-2020",
        "Telefon" : "",
        "TinggiBadan" : "",
        "golDarah" : "A-"
    }
},

"puffygmailcom" : {
    "Batman" : {
        "Name" : "Batman"
    },
    "Stitchh" : {
        "Name" : "Stitchh"
    }
}

3 个答案:

答案 0 :(得分:0)

一些技巧,也许是一个解决方案, 我建议使用firebase Cloud代替firebase实时firebase,它的速度更快且更可靠,特别是如果您尝试从数组或字典中查询数据,我可以看到您正在尝试从字典中检索数据,这是您要注意的一件事Firebase将您的swift字典存储为目标C字典,因此这就是您要注意的一件事!尝试检查您是否使用了正确的重用标识符。

让我知道您是否仍然无法获得它!

答案 1 :(得分:0)

如果您想返回数据列表,则必须使用.childAdded而不是.value

因此,您的代码将如下所示:

self.rootRef.child(self.user.emailWithoutSpecialChar).observe(.childAdded) { (snapshot) in

       // do your stuff here
}

* salam sesama orang Indonesia:)

答案 2 :(得分:0)

从问题的代码和结构看来,这里有医生和患者的列表,其中患者是医生的子节点。我将发布一个解决方案,然后发布一些有关更改的重要建议。

这是现有的Firebase结构。请注意,我们不使用电子邮件地址作为节点密钥-这是很多额外的工作,如果电子邮件地址发生更改,整个数据库将被扫描,节点读取,删除和重写。相反,应使用.childByAutoId生成动态节点密钥(可以更改的密钥,例如电子邮件)-我使用的是Doctor_0,doctor_1等,以提高可读性。

{
  "doctor_0" : {
    "name" : "Dr. Doolittle",
    "patient_list" : {
      "patient_0" : {
        "blood_type" : "O-",
        "name" : "Henry"
      },
      "patient_1" : {
        "blood_type" : "AB-",
        "name" : "Leroy"
      }
    }
  },
  "doctor_1" : {
    "name" : "Dr. McCoy",
    "patient_list" : {
      "patient_2" : {
        "blood_type" : "O+",
        "name" : "Steve"
      }
    }
  }
}

我有两个类来保存此数据,DoctorClass和PatientClass,其中Patient类是DoctorClass中的数组。请注意,医生可能没有任何患者,因此我将其视为可选项目。

class DoctorClass {
    var doc_id = ""
    var doc_name = ""
    var patients = [PatientClass]()
    
    convenience init(withId: String, andName: String, maybePatientList: DataSnapshot?) {
        self.init()
        self.doc_id = withId
        self.doc_name = andName
        
        if let patientList = maybePatientList {
            let allPatientsSnap = patientList.children.allObjects as! [DataSnapshot]
            for patientSnap in allPatientsSnap {
                let patient = PatientClass(patientSnap: patientSnap)
                self.patients.append(patient)
            }
        }
    }
}

class PatientClass {
    var patient_id = ""
    var patient_name = ""
    var blood_type = ""
    
    convenience init(patientSnap: DataSnapshot) {
        self.init()
        self.patient_id = patientSnap.key
        self.patient_name = patientSnap.childSnapshot(forPath: "name").value as? String ?? "No Name"
        self.blood_type = patientSnap.childSnapshot(forPath: "blood_type").value as? String ?? "No blood type"
    }
}

最后是要在所有医生中读取的代码,并用他们的患者填充他们的患者列表。

var docAndPatientList = [DoctorClass]()

func loadDoctorsAndPatients() {
    let doctorsRef = self.ref.child("doctors") //self.ref points to *my* firebase
    
    doctorsRef.observeSingleEvent(of: .value, with: { snapshot in
        let allDoctorsSnapshot = snapshot.children.allObjects as! [DataSnapshot]
        for docSnap in allDoctorsSnapshot {
            let docId = docSnap.key
            let docName = docSnap.childSnapshot(forPath: "name").value as? String ?? ""
            let patientSnap = docSnap.childSnapshot(forPath: "patient_list")
            let doc = DoctorClass(withId: docId, andName: docName, maybePatientList: patientSnap)
            self.docAndPatientList.append(doc)
        }
    })
}

然后将它们打印出来

func printDoctorsAndPatients() {
    self.docAndPatientList.forEach { doctor in
        print("Dr: \(doctor.doc_name)")
        for patient in doctor.patients {
            print("  patient: \(patient.patient_name)  bloodtype: \(patient.blood_type)")
        }
    }
}

和输出

Dr: Dr. Doolittle
  patient: Henry  bloodtype: O-
  patient: Leroy  bloodtype: AB-
Dr: Dr. McCoy
  patient: Steve  bloodtype: O+

这将适用于现有结构,但是例如,如果一个病人有两个医生,该怎么办?或者,如果我们想查询所有血型为O-的患者的Firebase,该怎么办?这种结构无法(轻松地)工作。

这是一个更好的计划

root_ref
   doctors
      doctor_0
         name : "Dr. Doolittle"
         patients
            patient_0 : true
            patient_1 : true
      doctor_1
         name : "Dr. McCoy"
         patients
            patient_2 : true

和患者

root_ref
   patients
      patient_0
         blood_type : "O-"
         doctors:
            doctor_0: true
         name : "Henry"
      patient_1
         blood_type : "AB-"
         doctors:
            doctor_0: true
         name : "Leroy"
      patient_2
         blood_type : "O+"
         doctors:
            doctor_1: true
         name : "Steve"

此结构提供了更多的查询灵活性和可伸缩性。