![](/img/trans.png)
[英]Checking if a particular value exists in a particular place in the firebase database
[英]Checking if a particular value exists in the Firebase database
我正在使用Firebase
实时数据库制作一个 Android 应用程序。 当新用户在我的应用上注册时,该用户的数据会保存在 Firebase 数据库中。
用户必须提供以下详细信息才能注册:
每当新用户尝试注册时,我必须确保每个用户的用户名都是唯一的,因此我会检查数据库中username
输入的用户username
是否已存在于数据库中。
为此,我编写了以下方法:
private boolean usernameExists(String username) {
DatabaseReference fdbRefer = FirebaseDatabase.getInstance().getReference("Users/"+username);
return (fdbRefer != null);
}
这个方法背后的逻辑是,如果getReference
方法找不到对指定路径的引用,它将返回 null 以便我可以返回fdbRefer
是否为null
。 如果fdbRefer
为null
,则表示数据库中不存在username
。
此方法的问题在于,无论输入的username
是否存在于数据库中,它始终返回true
。 这让我相信fdbRefer
永远不会为null
。
这让我想到了我的问题......
当getReference
方法在firebase
数据库中找不到指定路径时返回什么,以及检查username
已存在于数据库中的正确方法是什么?
要检查用户是否存在,请使用以下代码:
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);
您还可以使用 Query 来实现相同的功能:
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"
}
}
}
但由于在这种情况下,您的用户名已经是节点的名称,我建议您继续使用第一个。
尝试这个:
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
}
}
您必须使用orderbychild
和equalto
来检查值是否存在。 如果你不使用orderbychild
和equalto
那么它只会检查username
equalto
节点是否存在并且不关心值。
这个orderByChild("username").equalTo(Nick123)
就像在说:
WHERE username=Nick123
非常适合我
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) {
}
});
您可以使用 orderBy 查询来检查用户名是否已经存在,而不是检查引用是否存在
orderByChild('username').equalTo(username) 如果某些数据已经存在,则查询将返回数据,否则将返回空值。
像这样检查...
fdbRefer.addListenerForSingleValueEvent(new ValueEventListener() {
@Override
public void onDataChange(DataSnapshot dataSnapshot) {
if(dataSnapshot.exist() {
//username exist
}
else {
//username doesn't exist
}
}
});
由于您已经找到了问题的解决方案,我将尝试解释为什么您的代码不起作用。
DatabaseReference fdbRefer = FirebaseDatabase.getInstance().getReference("Users/"+username);
您可能会认为,如果数据库中不存在“Users/”+username,则在上述语句中fdbRefer
将为空。
但实际上,该语句只会在getReference()
存储指定字段的路径。 因此,当您检查fdbRef!=null
,它总是正确的,因为fdbRefer
将保存类似“https://.firebase.com/Users/Nick123”的值。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.