在我的应用程序中,我使用UserDefaults存储用户的登录状态(无论他们是否登录)及其用户名。它工作正常,当我登录时,关闭应用程序,然后再次打开它我的应用程序跳过登录页面并认识到我已经登录。虽然,我现在正在尝试将登出按钮安装到单独的viewController。单击时,此注销按钮需要1.)将UserDefaults.loginStatus重置为" False" 2.)将UserDefaults.username重置为nil 3.)对登录页面执行segue。
以下是我的ViewController.swift文件中的相关代码。这是第一个控制loginPage的viewController。
import UIKit
import Firebase
let defaults = UserDefaults.standard
class ViewController: UIViewController {
func DoLogin(username: String, password: String) {
//I Am not including a lot of the other stuff that takes place in this function, only the part that involves the defaults global variable
defaults.setValue(username, forKey: "username")
defaults.setValue("true", forKey: "loginStatus")
defaults.synchronize()
self.performSegue(withIdentifier: "loginToMain", sender: self) //This takes them to the main page of the app
}
override func viewDidLoad() {
super.viewDidLoad()
if let stringOne = defaults.string(forKey: "loginStatus") {
if stringOne == "true" { //If the user is logged in, proceed to main screen
DispatchQueue.main.async
{
self.performSegue(withIdentifier: "loginToMain", sender: self)
}
}
}
}
下面是我在SecondViewController.swift中的代码,特别是注销函数。
import UIKit
import Firebase
class SecondViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
if let username = defaults.string(forKey: "username") {
checkAppSetup(username: username) //This is an unrelated function
//I included this because this works fine. Proving that I am able to read the defaults variable fine from this other viewController
}
}
@IBAction func logout(_ sender: Any) {
defaults.setValue("false", forKey: "username")
defaults.setValue("false", forKey: "loginStatus")
defaults.synchronize()
performSegue(withIdentifier: "logoutSegue", sender: nil)
}
运行注销功能时,segue运行正常但默认值不会更改。有人可以解释为什么以及我能做些什么来解决这个问题?
**旁注,我实际上并没有将默认值设置为" false"和"假"。在调试此问题时,这只是暂时的。
答案 0 :(得分:2)
有几件事。
您应该使用set(_:forKey:)
和object(_:forKey
)来读取和写入默认值的键/值对,而不是setValue(_:forKey)
。 (但是,您对defaults.string(forKey: "loginStatus")
的使用是正确的。)
你应该写一个nil到userName键:
defaults.set(nil, forKey: "username")
您的退出IBAction
几乎肯定会将loginStatus
设为false
,而不是true
。
尝试改变这些事情。
此外,除非您在Xcode中终止应用程序而不是按设备/模拟器上的主页按钮以使其正常退出,否则没有理由调用同步。
答案 1 :(得分:2)
嘿,我最近使用了完全相同的概念:
1)在初始视图中,在viewDidLoad()
中,检查某人是否已经登录,并且一次只能有一个用户登录一个设备,所以我们检查
let defaults = UserDefaults.standard
if defaults.object(forKey: "userName") != nil && defaults.object(forKey: "userPassword") != nil
{
let loginObject = self.storyboard?.instantiateViewController(withIdentifier: "YourSecondViewController") as! YourSecondViewController
//As someone's details are already saved so we auto-login and move to second view
}}
2)在您的登录按钮功能中,检查您要检查的条件,然后在相同条件下,如果条件满足,则将数据保存到userDefaults。
// If no details are saved in defaults, then control will come to this part, where we will save the entered userName and Password
let defaults = UserDefaults.standard
defaults.set(self.enteredUseName, forKey: "userName")
defaults.set(self.enteredPassword, forKey: "Password")
defaults.synchronize()
3)在注销按钮上,删除userDefaults
并再次加载登录视图:
let defaults = UserDefaults.standard
defaults.removeObject(forKey: "userName") //We Will delete the userDefaults
defaults.removeObject(forKey: "userPassword")
defaults.synchronize() //Sync. the defaults.
navigationController?.popToRootViewController(animated: true) //Move Back to initial view.
4)如果您正在使用导航控件,那么您必须使用:P然后您肯定会看到后退按钮,如果单击该按钮将打开第二个视图,因为您可以隐藏{{1}中的导航栏您的登录视图
viewDidLoad()