我正在使用Firebase实时数据库制作一个Android应用程序。当新用户在我的应用上注册时,该用户的数据将保存在Firebase数据库中。

用户必须提供以下详细信息才能注册:


全名
电子邮件
用户名
密码

数据库结构


每当一个新用户尝试注册时,我都必须确保每个用户的用户名都是唯一的,因此我检查数据库是否用户输入的username是否已存在于数据库中。

为此,我编写了以下方法:

private boolean usernameExists(String username) {
     DatabaseReference fdbRefer = FirebaseDatabase.getInstance().getReference("Users/"+username);
     return (fdbRefer != null);
}


我的该方法背后的逻辑是,如果getReference方法找不到对指定路径的引用,它将返回null,以便我可以返回fdbRefer是否为null。如果fdbRefernull,则意味着数据库中不存在username

问题

此方法的问题是,它始终返回true输入的username是否存在是否在数据库中。这使我相信fdbRefer永远不会null

这使我想到我的问题...

问题

getReference方法在无法找到指定路径时返回什么? firebase数据库以及检查数据库中是否已经存在username的正确方法是什么?

#1 楼

要检查用户是否存在,请使用以下代码:
DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
DatabaseReference userNameRef = rootRef.child("Users").child("Nick123");
ValueEventListener eventListener = new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
        if(!dataSnapshot.exists()) {
            //create new user
        }
    }

    @Override
    public void onCancelled(DatabaseError databaseError) {
        Log.d(TAG, databaseError.getMessage()); //Don't ignore errors!
    }
};
userNameRef.addListenerForSingleValueEvent(eventListener);

您还可以使用查询来实现相同的功能:
DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
Query query = rootRef.child("Users").orderByChild("userName").equalTo("Nick123");
query.addValueEventListener(/* ... */);

这是另一种方法遍历整个Users节点,而不仅仅是使用对单个用户的直接引用。当您将uid用作用户而不是用户名时(如您现在所做的那样)用作唯一标识符时,更可能使用此选项。因此,如果您的数据库结构看起来像这样:
Firebase-root
   |
   --- Users
        |
        --- uid
             |
             --- userName: "Test User"
             |
             --- emailAddress: "user@email.com"

推荐使用第二种解决方案。
还有另一种解决方案,涉及创建另一个名为userNames的节点,在其中只能包含唯一的用户名。也请在下面找到相应的安全规则:
"Users": {
  "$uid": {
    ".write": "auth !== null && auth.uid === $uid",
    ".read": "auth !== null && auth.provider === 'password'",
    "userName": {
      ".validate": "
        !root.child('userNames').child(newData.val()).exists() ||
        root.child('userNames').child(newData.val()).val() == $uid"
    }
  }
}

但是由于在这种情况下,您的用户名已经是该节点的名称,所以我建议您继续使用第一个。

评论


另请参见这两种方法之间的区别。使用第一种解决方案将帮助您节省带宽。

– Alex Mamo
17年12月19日在19:34

您能解释一下if(!dataSnapshot.exists())的用途吗?

–user7500403
17/12/19在19:35



dataSnapshot对象表示以下位置的快照:Firebase-root-> Users-> Nick123。最好的解决方案是直接在dataSnapshot上使用exist()方法来检查是否存在。使用查询时,将遍历整个对象,这意味着浪费带宽。我还为您提供了查询解决方案,也许将来您会需要。

– Alex Mamo
17/12/19在19:44



@RedM第一种解决方案是使用直接指向特定用户的引用,第二种解决方案是使用查询整个对象以查找相应用户的查询。

– Alex Mamo
18/09/19在20:08

@MehulTank在这种情况下,您应该使用Peter的答案。

– Alex Mamo
19年8月13日在6:40

#2 楼

尝试以下操作:

DatabaseReference ref=FirebaseDatabase.getInstance().getReference().child("Users");
ref.orderByChild("username").equalTo(Nick123).addValueEventListener(new ValueEventListener(){
  @Override
  public void onDataChange(DataSnapshot dataSnapshot){
      if(dataSnapshot.exist() {
         //username exist
          }
        }


必须使用orderbychildequalto来检查值是否存在。
如果不使用orderbychildequalto,那么它将只需检查username子节点是否存在并且不关心该值。

orderByChild("username").equalTo(Nick123)就像说:

WHERE username=Nick123


#3 楼

非常适合我

DatabaseReference reference = FirebaseDatabase.getInstance().getReference();
        Query query = reference
                .child(getString(R.string.dbname_users))
                .orderByChild("username")
                .equalTo(username);
        query.addListenerForSingleValueEvent(new ValueEventListener() {
            @Override
            public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
                if(dataSnapshot.getChildrenCount()>0) {
                    //username found

                }else{
                    // username not found
                }

            }
            @Override
            public void onCancelled(DatabaseError databaseError) {

            }
        });


#4 楼

您可以使用orderBy查询来检查用户名是否已经存在,而不是检查引用是否存在。

orderByChild('username')。equalTo(username)查询将返回数据(如果已经存在其他数据)将返回null。

#5 楼

像这样检查...

    fdbRefer.addListenerForSingleValueEvent(new ValueEventListener() {
@Override
public void onDataChange(DataSnapshot dataSnapshot) {
if(dataSnapshot.exist() {
//username exist
}
else {
//username doesn't exist
}
}
});


#6 楼

由于您已经有了解决问题的方法,因此,我将尝试解释为什么您的代码无法正常工作。


 DatabaseReference fdbRefer = FirebaseDatabase.getInstance().getReference("Users/"+username);



您可能会请考虑在上面的语句中,如果数据库中不存在“ Users /” +用户名,则fdbRefer将为null。
但实际上,该语句将仅存储指向getReference()中指定字段的路径。因此,当您检查fdbRef!=null时,它始终是正确的,因为fdbRefer会保留类似'https://.firebase.com/Users/Nick123之类的值。