{
  "movies": {
    "movie1": {
      "genre": "comedy",
      "name": "As good as it gets",
      "lead": "Jack Nicholson"
    },
    "movie2": {
      "genre": "Horror",
      "name": "The Shining",
      "lead": "Jack Nicholson"
    },
    "movie3": {
      "genre": "comedy",
      "name": "The Mask",
      "lead": "Jim Carrey"
    }
  }
}

我是Firebase的新手。我如何从上面的数据中检索一个结果类型= '喜剧'和领导= '杰克尼克尔森'?

我有什么选择?


当前回答

Frank的回答很好,但是Firestore最近引入了数组包含,使得进行AND查询更容易。

您可以创建一个filters字段来添加筛选器。您可以根据需要添加任意多的值。例如,通过喜剧和杰克·尼克尔森过滤,你可以添加值comedy_杰克·尼克尔森,但如果你也想通过喜剧和2014,你可以添加值comedy_2014,而不需要创建更多字段。

{
  "movies": {
    "movie1": {
      "genre": "comedy",
      "name": "As good as it gets",
      "lead": "Jack Nicholson",
      "year": 2014,
      "filters": [
        "comedy_Jack Nicholson",
        "comedy_2014"
      ]
    }
  }
}

其他回答

Frank的回答很好,但是Firestore最近引入了数组包含,使得进行AND查询更容易。

您可以创建一个filters字段来添加筛选器。您可以根据需要添加任意多的值。例如,通过喜剧和杰克·尼克尔森过滤,你可以添加值comedy_杰克·尼克尔森,但如果你也想通过喜剧和2014,你可以添加值comedy_2014,而不需要创建更多字段。

{
  "movies": {
    "movie1": {
      "genre": "comedy",
      "name": "As good as it gets",
      "lead": "Jack Nicholson",
      "year": 2014,
      "filters": [
        "comedy_Jack Nicholson",
        "comedy_2014"
      ]
    }
  }
}
ref.orderByChild("lead").startAt("Jack Nicholson").endAt("Jack Nicholson").listner....

这是可行的。

我已经编写了一个个人库,允许您按多个值进行排序,所有排序都在服务器上完成。

见到Querybase !

Querybase接受一个Firebase数据库引用和一个字段数组,你想在上面建立索引。当您创建新记录时,它将自动处理允许多次查询的键的生成。需要注意的是,它只支持直接等价(不小于或大于)。

const databaseRef = firebase.database().ref().child('people');
const querybaseRef = querybase.ref(databaseRef, ['name', 'age', 'location']);

// Automatically handles composite keys
querybaseRef.push({ 
  name: 'David',
  age: 27,
  location: 'SF'
});

// Find records by multiple fields
// returns a Firebase Database ref
const queriedDbRef = querybaseRef
 .where({
   name: 'David',
   age: 27
 });

// Listen for realtime updates
queriedDbRef.on('value', snap => console.log(snap));

Firebase不允许使用多个条件进行查询。 然而,我确实找到了一个解决方法:

我们需要从数据库下载初始过滤数据,并将其存储在一个数组列表中。

                Query query = databaseReference.orderByChild("genre").equalTo("comedy");
                databaseReference.addValueEventListener(new ValueEventListener() {
                    @Override
                    public void onDataChange(@NonNull DataSnapshot dataSnapshot) {

                        ArrayList<Movie> movies = new ArrayList<>();
                        for (DataSnapshot dataSnapshot1 : dataSnapshot.getChildren()) {
                            String lead = dataSnapshot1.child("lead").getValue(String.class);
                            String genre = dataSnapshot1.child("genre").getValue(String.class);

                            movie = new Movie(lead, genre);

                            movies.add(movie);

                        }

                        filterResults(movies, "Jack Nicholson");

                        }

                    }

                    @Override
                    public void onCancelled(@NonNull DatabaseError databaseError) {

                    }
                });

一旦我们从数据库中获得了初始过滤数据,我们需要在后端进行进一步的过滤。

public void filterResults(final List<Movie> list,  final String genre) {
        List<Movie> movies = new ArrayList<>();
        movies = list.stream().filter(o -> o.getLead().equals(genre)).collect(Collectors.toList());
        System.out.println(movies);

        employees.forEach(movie -> System.out.println(movie.getFirstName()));
    }

对于云壁炉

https://firebase.google.com/docs/firestore/query-data/queries#compound_queries

复合查询 您可以将多个相等操作符(==或数组包含)方法链接起来,以创建更具体的查询(逻辑与)。但是,您必须创建一个复合索引来将相等操作符与不等操作符、<、<=、>和!=组合在一起。

citiesRef.where('state', '==', 'CO').where('name', '==', 'Denver');
citiesRef.where('state', '==', 'CA').where('population', '<', 1000000);

你可以只在一个字段上执行range(<, <=, >, >=)或not equals(!=)比较,并且在复合查询中你最多可以包含一个array-contains或array-contains-any子句: