使用onDataChange验证值的解决方案(Firebase数据库和Firebase身份验证)

时间:2019-01-25 18:18:17

标签: android firebase android-studio kotlin

我试图使用用户的角色来验证身份,所以我设置了一个布尔值以检查用户的角色是“培训师”还是“培训生”。但是onDataChange仅在意图之后才更改布尔值...

private lateinit var auth: FirebaseAuth
private lateinit var firebaseDatabase: FirebaseDatabase
private lateinit var databaseReference: DatabaseReference
var trainer:Boolean = false

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_login)
    auth = FirebaseAuth.getInstance()
    firebaseDatabase = FirebaseDatabase.getInstance()
    databaseReference = firebaseDatabase.getReference("users")

    backRegister.setOnClickListener {
        val intent = Intent(this, MainActivity::class.java)
        startActivity(intent)
    }

    btnSignIn.setOnClickListener {
        val email = etEmail.text.toString()
        val pass = etPass.text.toString()
        login(email, pass)
    }
}

fun login(email:String, pass:String){
    auth.signInWithEmailAndPassword(email, pass)
        .addOnCompleteListener(this) { task ->
            if (task.isSuccessful) {
                val user = auth.currentUser
                if (user != null) {
                    checkRole(user)
                    if(trainer){
                        intent = Intent(this, HomeTrainer::class.java)
                        Toast.makeText(baseContext, "Welcome Trainer",
                            Toast.LENGTH_SHORT).show()
                        startActivity(intent)
                    }
                    else {
                        var intent = Intent(this, Home::class.java)
                        Toast.makeText(baseContext, "Welcome Trainee",
                            Toast.LENGTH_SHORT).show()
                        startActivity(intent)
                    }
                }
            } else {
                Toast.makeText(baseContext, "Login Failed",
                    Toast.LENGTH_SHORT).show()
            }
        }
}

fun checkRole(user:FirebaseUser){
    val postListener = object : ValueEventListener {
        override fun onDataChange(dataSnapshot: DataSnapshot) {
            for (datas in dataSnapshot.children) {
                if (datas.key.toString() == user.uid) {
                    trainer = false
                    if (datas.child("role").getValue().toString().trim().equals("trainer")) {
                        trainer = true
                    }
                }
            }
        }
        override fun onCancelled(databaseError: DatabaseError) {
        }
    }
    databaseReference.addValueEventListener(postListener)
}

1 个答案:

答案 0 :(得分:1)

Firebase API为asynchronous,这意味着onDataChange()函数在调用后立即返回,并且它返回的任务的回调将在一段时间后调用。无法保证需要多长时间。因此,可能需要几百毫秒到几秒钟的时间才能获得该数据。由于该方法会立即返回,因此您尚未尝试从回调中填充您要使用的trainer变量的值。因此,简单地将其创建为全局变量将无济于事,其值始终为false

基本上,您正在尝试从异步的API同步使用值。那不是一个好主意。您应该按预期异步处理API。

此问题的快速解决方案是在回调中仅使用您的bool变量 的值,否则我建议您从此 post ,其中我解释了如何使用自定义回调来完成。