All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.lucene.search.grouping.package-info Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * Grouping.
 *
 * 

This module enables search result grouping with Lucene, where hits with the same value in the * specified single-valued group field are grouped together. For example, if you group by the * author field, then all documents with the same value in the author field fall * into a single group. * *

Grouping requires a number of inputs: * *

    *
  • groupSelector: this defines how groups are created from values per-document. * The grouping module ships with selectors for grouping by term, and by long and double * ranges. *
  • groupSort: how the groups are sorted. For sorting purposes, each group is * "represented" by the highest-sorted document according to the groupSort within * it. For example, if you specify "price" (ascending) then the first group is the one with * the lowest price book within it. Or if you specify relevance group sort, then the first * group is the one containing the highest scoring book. *
  • topNGroups: how many top groups to keep. For example, 10 means the top 10 * groups are computed. *
  • groupOffset: which "slice" of top groups you want to retrieve. For example, 3 * means you'll get 7 groups back (assuming topNGroups is 10). This is useful for * paging, where you might show 5 groups per page. *
  • withinGroupSort: how the documents within each group are sorted. This can be * different from the group sort. *
  • maxDocsPerGroup: how many top documents within each group to keep. *
  • withinGroupOffset: which "slice" of top documents you want to retrieve from * each group. *
* *

The implementation is two-pass: the first pass ({@link * org.apache.lucene.search.grouping.FirstPassGroupingCollector}) gathers the top groups, and the * second pass ({@link org.apache.lucene.search.grouping.SecondPassGroupingCollector}) gathers * documents within those groups. If the search is costly to run you may want to use the {@link * org.apache.lucene.search.CachingCollector} class, which caches hits and can (quickly) replay them * for the second pass. This way you only run the query once, but you pay a RAM cost to (briefly) * hold all hits. Results are returned as a {@link org.apache.lucene.search.grouping.TopGroups} * instance. * *

Groups are defined by {@link org.apache.lucene.search.grouping.GroupSelector} implementations: * *

    *
  • {@link org.apache.lucene.search.grouping.TermGroupSelector} groups based on the value of a * {@link org.apache.lucene.index.SortedDocValues} field *
  • {@link org.apache.lucene.search.grouping.ValueSourceGroupSelector} groups based on the * value of a {@link org.apache.lucene.queries.function.ValueSource} *
  • {@link org.apache.lucene.search.grouping.DoubleRangeGroupSelector} groups based on the * value of a {@link org.apache.lucene.search.DoubleValuesSource} *
  • {@link org.apache.lucene.search.grouping.LongRangeGroupSelector} groups based on the value * of a {@link org.apache.lucene.search.LongValuesSource} *
* *

Known limitations: * *

    *
  • Sharding is not directly supported, though is not too difficult, if you can merge the top * groups and top documents per group yourself. *
* *

Typical usage for the generic two-pass grouping search looks like this using the grouping * convenience utility (optionally using caching for the second pass search): * *

 *   GroupingSearch groupingSearch = new GroupingSearch("author");
 *   groupingSearch.setGroupSort(groupSort);
 *   groupingSearch.setFillSortFields(fillFields);
 *
 *   if (useCache) {
 *     // Sets cache in MB
 *     groupingSearch.setCachingInMB(4.0, true);
 *   }
 *
 *   if (requiredTotalGroupCount) {
 *     groupingSearch.setAllGroups(true);
 *   }
 *
 *   TermQuery query = new TermQuery(new Term("content", searchTerm));
 *   TopGroups<BytesRef> result = groupingSearch.search(indexSearcher, query, groupOffset, groupLimit);
 *
 *   // Render groupsResult...
 *   if (requiredTotalGroupCount) {
 *     int totalGroupCount = result.totalGroupCount;
 *   }
 * 
* *

To use the single-pass BlockGroupingCollector, first, at indexing time, you must * ensure all docs in each group are added as a block, and you have some way to find the last * document of each group. One simple way to do this is to add a marker binary field: * *

 *   // Create Documents from your source:
 *   List<Document> oneGroup = ...;
 *
 *   Field groupEndField = new Field("groupEnd", "x", Field.Store.NO, Field.Index.NOT_ANALYZED);
 *   groupEndField.setIndexOptions(IndexOptions.DOCS_ONLY);
 *   groupEndField.setOmitNorms(true);
 *   oneGroup.get(oneGroup.size()-1).add(groupEndField);
 *
 *   // You can also use writer.updateDocuments(); just be sure you
 *   // replace an entire previous doc block with this new one.  For
 *   // example, each group could have a "groupID" field, with the same
 *   // value for all docs in this group:
 *   writer.addDocuments(oneGroup);
 * 
* * Then, at search time: * *
 *   Query groupEndDocs = new TermQuery(new Term("groupEnd", "x"));
 *   BlockGroupingCollector c = new BlockGroupingCollector(groupSort, groupOffset+topNGroups, needsScores, groupEndDocs);
 *   s.search(new TermQuery(new Term("content", searchTerm)), c);
 *   TopGroups groupsResult = c.getTopGroups(withinGroupSort, groupOffset, docOffset, docOffset+docsPerGroup, fillFields);
 *
 *   // Render groupsResult...
 * 
* * Or alternatively use the GroupingSearch convenience utility: * *
 *   // Per search:
 *   GroupingSearch groupingSearch = new GroupingSearch(groupEndDocs);
 *   groupingSearch.setGroupSort(groupSort);
 *   groupingSearch.setIncludeScores(needsScores);
 *   TermQuery query = new TermQuery(new Term("content", searchTerm));
 *   TopGroups groupsResult = groupingSearch.search(indexSearcher, query, groupOffset, groupLimit);
 *
 *   // Render groupsResult...
 * 
* * Note that the groupValue of each GroupDocs will be null, * so if you need to present this value you'll have to separately retrieve it (for example using * stored fields, FieldCache, etc.). * *

Another collector is the AllGroupHeadsCollector that can be used to retrieve all * most relevant documents per group. Also known as group heads. This can be useful in situations * when one wants to compute group based facets / statistics on the complete query result. The * collector can be executed during the first or second phase. This collector can also be used with * the GroupingSearch convenience utility, but when if one only wants to compute the * most relevant documents per group it is better to just use the collector as done here below. * *

 *   TermGroupSelector grouper = new TermGroupSelector(groupField);
 *   AllGroupHeadsCollector c = AllGroupHeadsCollector.newCollector(grouper, sortWithinGroup);
 *   s.search(new TermQuery(new Term("content", searchTerm)), c);
 *   // Return all group heads as int array
 *   int[] groupHeadsArray = c.retrieveGroupHeads()
 *   // Return all group heads as FixedBitSet.
 *   int maxDoc = s.maxDoc();
 *   FixedBitSet groupHeadsBitSet = c.retrieveGroupHeads(maxDoc)
 * 
*/ package org.apache.lucene.search.grouping;




© 2015 - 2025 Weber Informatics LLC | Privacy Policy