- I have an app that a user registers animals,
- these animals are separated by lot,
- so, to know where the animal is, the user just types his TAG and I look for the tag in all lots of my database,
- example:
Collection> Lots
Documents: Lot+NumberLot (Lot+number is the ID, for example: Lot01, Lot02).
Within each Document Lot+NumberLot I have a subcollection “Animals”, within each subCollection animals,
I have documents for each animal, where the ID is the TAG that the user types.
What I do is as follows, the user types the TAG but he does not know the lot, so, I search for that TAG in all lots, however, if the user has many lots and wants to search only 1 animal, I will be charged for all lots even if it is to retrieve a single document from “Animals”, I am currently doing this:
-My Code
lotRef.get()
.addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
@Override
public void onComplete(@NonNull Task<QuerySnapshot> task) {
if(task.isSuccessful()){
for (QueryDocumentSnapshot documentSnapshot: task.getResult()){
CollectionReference aniRef = lotRef.document(documentSnapshot.getId().toString())
.collection("Animals");
aniRef.get()
.addOnCompleteListener(new OnCompleteListener<QuerySnapshot>() {
@Override
public void onComplete(@NonNull Task<QuerySnapshot> task) {
if(task.isSuccessful()) {
for (DocumentSnapshot documentSnapshot1 : task.getResult()) {
Animal aniFounded= documentSnapshot1.toObject(Animal.class);
if (aniFounded.getTagAni() != null && aniFounded.getTagAni().equals(tagAni)) {
Toast.makeText(Activity_AttSemBLE.this, aniFounded.getTagBoi().toString(),
Toast.LENGTH_LONG).show();
founded = true;
break;
}
}
}else{
Toast.makeText(Activity_AttSemBLE.this, "Not found", Toast.LENGTH_LONG).show();
}
}
});
if(founded){
founded = false;
break;
}
}
}else{
Toast.makeText(Activity_AttSemBLE.this, "No animal found", Toast.LENGTH_LONG).show();
}
}
});
This program finds the animal if it exists, but I’m always reading all the documents and all the lots, I wanted to make sure that after I found the animal, I would not be charged any more, if I use Where will I still be charged for all documents?
For example, I want to find the 041712 TAG, but I don’t know the lot it is in (since the user will not remember the lot, only the Tag), I didn’t want to have to go through all the batches and tags to find it, I wanted that after finding him, I don’t need to read the rest so I would not consume unnecessary queries
Advertisement
Answer
You have an animal object that has as the id of the document 041712
. If this id is also contained as a property within that document, to get all the animals without knowing the lot, a collection group query is required. In code, it should look similar to this:
db.collectionGroup("Animals").whereEqualTo("id", "041712").limit(1).get()
.addOnSuccessListener(new OnSuccessListener<QuerySnapshot>() {
@Override
public void onSuccess(QuerySnapshot queryDocumentSnapshots) {
// Iterate the queryDocumentSnapshots object
}
});