全局缓存服务器数据并刷新视图
Cache server data globally and refresh views
我正在构建一个使用 firebase 进行身份验证和数据库功能的应用程序。一旦用户注册,就会为该用户存储一条数据库记录,其中包含一些基本信息,如名字、姓氏等。
一旦用户使用他的凭据登录,我想设置一个全局变量(也许是 userDefaults?),其中包含该特定用户的用户数据。否则,每次我想用例如用户的名字填充标签时,我都必须获取用户数据。
我设法在登录时设置用户默认值并在 UIlables 中使用此信息。但是当我让用户更改他们的数据时,其中一些对应用程序的功能很重要,我可以更新服务器和用户默认值,但应用程序本身不会更新正确的数据。它将旧数据保存在(例如)UIlables 中。
我很想更深入地了解什么是管理此类情况的最佳工作流程。
打开应用程序时,我将 tabBarController
设置为 rootviewcontroller
。在 tabbarcontroller
的负载中,我有以下代码从 firebase 检索用户数据并将其保存到 userdefaults
:
guard let uid = Auth.auth().currentUser?.uid else { return }
Database.database().reference().child("users").child(uid).observeSingleEvent(of: .value, with: { (snapshot) in
print(snapshot.value ?? "")
guard let dictionary = snapshot.value as? [String: Any] else { return }
let firstname = dictionary["First name"] as? String
let lastname = dictionary["Last name"] as? String
print("first name is: " + firstname!)
UserDefaults.standard.set(firstname, forKey: "userFirstName")
print(UserDefaults.standard.value(forKey: "userFirstName"))
self.setupViewControllers()
}
然后我继续加载 tabBarController 中的所有视图控制器:
self.setupViewControllers()
在该过程中,这些视图控制器中的标签会填充用户默认数据。
这是一个用 userDefaults 填充但在更改 userdefaults 时未更新的标签示例:
let welcomeLabel: UILabel = {
let label = UILabel()
let attributedText = NSMutableAttributedString(string: "Welcome ")
attributedText.append(NSAttributedString(string: "\(UserDefaults.standard.string(forKey: "userFirstName")!)"))
label.attributedText = attributedText
label.font = UIFont.systemFont(ofSize: 30, weight: .bold)
return label
}()
这是我用来更新名字的函数(通过用户填写的文本字段):
@objc func updateName() {
guard let uid = Auth.auth().currentUser?.uid else { return }
Database.database().reference().child("users").child(uid).updateChildValues(["First name" : updateNameField.text ?? ""])
UserDefaults.standard.set(updateNameField.text, forKey: "userFirstName")
print(UserDefaults.standard.value(forKey: "userFirstName"))
}
所以你必须先整理一下。在新文件中定义如下常量。除非 private
,否则这些常量将在全局范围内可访问
Constants.swift
private let storedusername = "usname"
private let storedName = "uname"
private let displaypic = "udp"
private let aboutme = "udesc"
var myusername : String {
get {
return (UserDefaults.standard.string(forKey: storedusername)!)
} set {
UserDefaults.standard.set(newValue, forKey: storedusername)
}
}
var myname : String {
get {
return (UserDefaults.standard.string(forKey: storedName)!)
} set {
UserDefaults.standard.set(newValue, forKey: storedName)
}
}
var myProfileImage : Data {
get {
return (UserDefaults.standard.data(forKey: displaypic)!)
} set {
UserDefaults.standard.set(newValue, forKey: displaypic)
}
}
var myAboutMe : String? {
get {
return (UserDefaults.standard.string(forKey: aboutme)!)
} set {
UserDefaults.standard.set(newValue, forKey: aboutme)
}
}
现在,下次您想在 UserDefaults
中保存任何内容时,只需在整个代码库的任何位置执行以下操作即可:
myusername = "@CVEIjk"
要检索它,只需调用它:
print(myusername)
重要提示 --
永远记得初始化它们。您可以在用户注册时执行此操作。一旦他们填写了详细信息并点击提交,就将它们保存到这些变量中。那不会造成不必要的崩溃。
您必须在数据库中对这些节点执行更新的每个位置保存它们。
现在,刷新视图部分。我正在考虑一个场景,您的 ProfileView.swift
有视图,用户转到 EditProfile.swift
更新内容。
您在更新将立即生效的地方初始化所有观察者。因为更新后的视图很重要。剩下的会通过aboutme
的getter调用
ProfileView.swift
func openEditView() {
NotificationCenter.default.addObserver(self, selector: #selector(fetchUserDetails), name: Notification.Name("update"), object: nil)
//setting this right before the segue will create an observer specifically and exclusively for updates. Hence you don't have to worry about the extra observers.
perform(segue: With Identifier:)// Goes to the editProfile page
}
该函数最初将在 viewDidLoad() 中调用。这时候你需要确保你拥有所有的数据,否则它不会产生任何值。但是,如果您在用户注册时存储所有内容,那么您是安全的。
@objc func fetchUserDetails() {
if uid != nil {
if myname.count > 0 { // This will check if the variable has anything in the memory or not. Dont confuse this with [Array].count
self.nameLabel = myname
}
}
}
这个函数也是一个ab观察者方法。因此,当发布通知时,他们可以再次 运行。
现在,EditProfile.swift
在更新服务器的块中,保存值,然后创建一个 Notification.post
并将此方法放在 dismiss(toViewController:)
之前
func updateUserCacheData(name: String, username: String, aboutme: String, ProfilePhoto: UIImage? = nil) {
DispatchQueue.global().async {
myname = name
myusername = username
myAboutMe = aboutme
if self.newImage != nil {
myProfileImage = self.newImage!.jpegData(compressionQuality: 1)!
}
DispatchQueue.main.async {
NotificationCenter.default.post(name: .refreshProfileViews, object: nil)
}
}
}
func updateToServerAndBackToProfileView() {
self.updateUserCacheData(name: iname!, username: iusername, aboutme: iaboutme!)
self.dismiss(animated: true, completion: nil)
}
}
只要回到ProfileView
,你的观点就会瞬间刷新。你可以保留一个观察者,无论你在哪里查看都会在解散后首先显示。其余的将始终获取更新的内容。另外,不要忘记 deinit
你的观察者 ProfileView
//This code is in ProfileView.swift
deinit {
NotificationCenter.default.removeObserver(self, name: Notification.Name("update"), object: nil)
}
此外,在内容可能为空的情况下,只需使用空内容对其进行初始化即可。例如,如果用户在注册时没有选择添加关于我,你可以直接输入
`myaboutme = ""`
这将为您创造一个安全的环境,您已经准备就绪。
我正在构建一个使用 firebase 进行身份验证和数据库功能的应用程序。一旦用户注册,就会为该用户存储一条数据库记录,其中包含一些基本信息,如名字、姓氏等。
一旦用户使用他的凭据登录,我想设置一个全局变量(也许是 userDefaults?),其中包含该特定用户的用户数据。否则,每次我想用例如用户的名字填充标签时,我都必须获取用户数据。
我设法在登录时设置用户默认值并在 UIlables 中使用此信息。但是当我让用户更改他们的数据时,其中一些对应用程序的功能很重要,我可以更新服务器和用户默认值,但应用程序本身不会更新正确的数据。它将旧数据保存在(例如)UIlables 中。
我很想更深入地了解什么是管理此类情况的最佳工作流程。
打开应用程序时,我将 tabBarController
设置为 rootviewcontroller
。在 tabbarcontroller
的负载中,我有以下代码从 firebase 检索用户数据并将其保存到 userdefaults
:
guard let uid = Auth.auth().currentUser?.uid else { return }
Database.database().reference().child("users").child(uid).observeSingleEvent(of: .value, with: { (snapshot) in
print(snapshot.value ?? "")
guard let dictionary = snapshot.value as? [String: Any] else { return }
let firstname = dictionary["First name"] as? String
let lastname = dictionary["Last name"] as? String
print("first name is: " + firstname!)
UserDefaults.standard.set(firstname, forKey: "userFirstName")
print(UserDefaults.standard.value(forKey: "userFirstName"))
self.setupViewControllers()
}
然后我继续加载 tabBarController 中的所有视图控制器:
self.setupViewControllers()
在该过程中,这些视图控制器中的标签会填充用户默认数据。
这是一个用 userDefaults 填充但在更改 userdefaults 时未更新的标签示例:
let welcomeLabel: UILabel = {
let label = UILabel()
let attributedText = NSMutableAttributedString(string: "Welcome ")
attributedText.append(NSAttributedString(string: "\(UserDefaults.standard.string(forKey: "userFirstName")!)"))
label.attributedText = attributedText
label.font = UIFont.systemFont(ofSize: 30, weight: .bold)
return label
}()
这是我用来更新名字的函数(通过用户填写的文本字段):
@objc func updateName() {
guard let uid = Auth.auth().currentUser?.uid else { return }
Database.database().reference().child("users").child(uid).updateChildValues(["First name" : updateNameField.text ?? ""])
UserDefaults.standard.set(updateNameField.text, forKey: "userFirstName")
print(UserDefaults.standard.value(forKey: "userFirstName"))
}
所以你必须先整理一下。在新文件中定义如下常量。除非 private
Constants.swift
private let storedusername = "usname"
private let storedName = "uname"
private let displaypic = "udp"
private let aboutme = "udesc"
var myusername : String {
get {
return (UserDefaults.standard.string(forKey: storedusername)!)
} set {
UserDefaults.standard.set(newValue, forKey: storedusername)
}
}
var myname : String {
get {
return (UserDefaults.standard.string(forKey: storedName)!)
} set {
UserDefaults.standard.set(newValue, forKey: storedName)
}
}
var myProfileImage : Data {
get {
return (UserDefaults.standard.data(forKey: displaypic)!)
} set {
UserDefaults.standard.set(newValue, forKey: displaypic)
}
}
var myAboutMe : String? {
get {
return (UserDefaults.standard.string(forKey: aboutme)!)
} set {
UserDefaults.standard.set(newValue, forKey: aboutme)
}
}
现在,下次您想在 UserDefaults
中保存任何内容时,只需在整个代码库的任何位置执行以下操作即可:
myusername = "@CVEIjk"
要检索它,只需调用它:
print(myusername)
重要提示 --
永远记得初始化它们。您可以在用户注册时执行此操作。一旦他们填写了详细信息并点击提交,就将它们保存到这些变量中。那不会造成不必要的崩溃。
您必须在数据库中对这些节点执行更新的每个位置保存它们。
现在,刷新视图部分。我正在考虑一个场景,您的 ProfileView.swift
有视图,用户转到 EditProfile.swift
更新内容。
您在更新将立即生效的地方初始化所有观察者。因为更新后的视图很重要。剩下的会通过aboutme
ProfileView.swift
func openEditView() {
NotificationCenter.default.addObserver(self, selector: #selector(fetchUserDetails), name: Notification.Name("update"), object: nil)
//setting this right before the segue will create an observer specifically and exclusively for updates. Hence you don't have to worry about the extra observers.
perform(segue: With Identifier:)// Goes to the editProfile page
}
该函数最初将在 viewDidLoad() 中调用。这时候你需要确保你拥有所有的数据,否则它不会产生任何值。但是,如果您在用户注册时存储所有内容,那么您是安全的。
@objc func fetchUserDetails() {
if uid != nil {
if myname.count > 0 { // This will check if the variable has anything in the memory or not. Dont confuse this with [Array].count
self.nameLabel = myname
}
}
}
这个函数也是一个ab观察者方法。因此,当发布通知时,他们可以再次 运行。
现在,EditProfile.swift
在更新服务器的块中,保存值,然后创建一个 Notification.post
并将此方法放在 dismiss(toViewController:)
func updateUserCacheData(name: String, username: String, aboutme: String, ProfilePhoto: UIImage? = nil) {
DispatchQueue.global().async {
myname = name
myusername = username
myAboutMe = aboutme
if self.newImage != nil {
myProfileImage = self.newImage!.jpegData(compressionQuality: 1)!
}
DispatchQueue.main.async {
NotificationCenter.default.post(name: .refreshProfileViews, object: nil)
}
}
}
func updateToServerAndBackToProfileView() {
self.updateUserCacheData(name: iname!, username: iusername, aboutme: iaboutme!)
self.dismiss(animated: true, completion: nil)
}
}
只要回到ProfileView
,你的观点就会瞬间刷新。你可以保留一个观察者,无论你在哪里查看都会在解散后首先显示。其余的将始终获取更新的内容。另外,不要忘记 deinit
你的观察者 ProfileView
//This code is in ProfileView.swift
deinit {
NotificationCenter.default.removeObserver(self, name: Notification.Name("update"), object: nil)
}
此外,在内容可能为空的情况下,只需使用空内容对其进行初始化即可。例如,如果用户在注册时没有选择添加关于我,你可以直接输入
`myaboutme = ""`
这将为您创造一个安全的环境,您已经准备就绪。