ios - 全局缓存服务器数据并刷新 View

标签 ios swift firebase userdefaults

我正在构建一个使用 firebase 进行身份验证和数据库功能的应用。用户注册后,就会为该用户存储一条数据库记录,其中包含一些基本信息,例如名字、姓氏等。

用户使用其凭据登录后,我想设置一个全局变量(可能是 userDefaults?),其中包含该特定用户的用户数据。否则,每次我想填充标签时,我都必须获取用户数据,例如用户的名字。

我设法在登录时设置用户默认值并在 UIlables 中使用此信息。但是,当我让用户更改其数据(其中一些数据对于应用程序的功能很重要)时,我可以更新服务器和用户默认值,但应用程序本身不会使用正确的数据进行更新。它将旧数据保存在(例如)UIlable 中。

我很想更深入地了解管理此类情况的最佳工作流程。

打开应用程序时,我将 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 中的所有 View Controller :

self.setupViewControllers()

在此过程中,这些 View Controller 中的标签将被用户默认数据填充。

这是一个使用 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)

重要说明--

  1. 永远记住初始化它们。您可以在用户注册时执行此操作。一旦他们填写了详细信息并点击提交,只需将它们保存到这些变量中即可。这不会导致不必要的崩溃。

  2. 您必须将它们保存在数据库中对这些节点执行更新的每个位置。

现在,刷新 View 部分。我正在考虑这样一个场景,您的 ProfileView.swift 具有 View ,并且用户转到 EditProfile.swift 更新内容。

您初始化所有观察者,更新将立即生效。因为更新后的 View 很重要。其余的将通过 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

在要更新服务器的 block 中,保存值,然后创建一个 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,您的 View 就会立即刷新。您可以在您查看的任何地方保留一个观察员,该观察员将在解散后首先显示。其余的将始终获取更新的内容。另外,不要忘记在 ProfileViewdeinit 你的观察者

//This code is in ProfileView.swift
deinit {
        NotificationCenter.default.removeObserver(self, name: Notification.Name("update"), object: nil)
    }

此外,如果内容可能为空,只需使用空内容对其进行初始化即可。例如,如果用户在注册时没有选择添加 aboutme,您可以直接输入

`myaboutme = ""`

这将为您创造一个安全的环境,您已做好准备。

关于ios - 全局缓存服务器数据并刷新 View ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56373130/

相关文章:

ios - WKWebView媒体控件标题显示Url

ios - Swift 如何添加工作时钟?

swift - Firebase 自定义 reCaptach AuthUIDelegate - Swift

flutter - GoogleAuthProvider - 选择帐户的选项

iphone - 在 UIImageView 中旋转图像

ios - 我想用 UnsafeMutableRawPointer 偏移量快速编写代码

ios - 在 UITextField 操作中将 UITextField 字符串转换为 Double

ios - 如何在 Swift 中使用函数更改标签后将其更改为原始文本?

ios - 从 UIView 捕获 UIImage 不工作

javascript - 如何映射和减少 Firebase List Observable 的子值