全局缓存服务器数据并刷新视图

Posted

技术标签:

【中文标题】全局缓存服务器数据并刷新视图【英文标题】:Cache server data globally and refresh views 【发布时间】:2019-05-30 06:49:57 【问题描述】:

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

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

我设法在登录时设置 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"))
    

【问题讨论】:

这看起来像是您需要使用通知实现 Observable 模式的情况,以便在保存新/更新的数据时,应用程序的其他部分会收到有关它的通知并可以自行更新。例如见this article @CvEijk 您希望存储哪些信息。你能说得更具体些吗,我可以帮忙 一个更简单的选择是将使用视图控制器中的用户信息更新 UI 的所有代码收集到一个方法中,这样当您保存用户数据时,只需调用该方法即可。类似于tableView.reloadData() @JoakimDanielson 他需要一些东西来刷新视图。这只是设置您提到的观察者并刷新通知视图。但他是否需要更多。如果是这种情况,将使用更好的缓存系统。顺便说一句,两者都很容易实现。我们需要了解他的要求 @AakashDave 这已经困扰我太久了哈哈,感谢您花时间和精力分享您的专业知识,非常感谢:) 【参考方案1】:

所以你必须先把事情整理好。在一个新文件中定义如下常量。这些常量可以在全局范围内访问,除非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)
    

此外,在内容可能为空的情况下,只需使用空内容对其进行初始化即可。例如,如果用户在注册时没有选择添加 aboutme,则可以添加

`myaboutme = ""`

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

【讨论】:

以上是关于全局缓存服务器数据并刷新视图的主要内容,如果未能解决你的问题,请参考以下文章

如何告诉 OkHttpClient 忽略缓存并从服务器强制刷新?

单例设计模式全局缓存accessToken

Apollo缓存+自动刷新

页面刷新的一些方法

Laravel 视图显示在一台服务器上的缓存中,但在另一台服务器上工作正常

Django模板缓存刷新问题