
org.apache.lucene.document.DoubleField Maven / Gradle / Ivy
Show all versions of org.apache.servicemix.bundles.lucene
/*
* 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.
*/
package org.apache.lucene.document;
import org.apache.lucene.index.DocValuesType;
import org.apache.lucene.index.PointValues;
import org.apache.lucene.search.IndexOrDocValuesQuery;
import org.apache.lucene.search.PointRangeQuery;
import org.apache.lucene.search.Query;
import org.apache.lucene.search.SortField;
import org.apache.lucene.search.SortedNumericSelector;
import org.apache.lucene.search.SortedNumericSortField;
import org.apache.lucene.util.BytesRef;
import org.apache.lucene.util.NumericUtils;
/**
* Field that stores a per-document double
value for scoring, sorting or value
* retrieval and index the field for fast range filters. If you need more fine-grained control you
* can use {@link DoublePoint}, {@link DoubleDocValuesField} and {@link StoredField}.
*
* This field defines static factory methods for creating common queries:
*
*
* - {@link #newExactQuery(String, double)} for matching an exact 1D point.
*
- {@link #newRangeQuery(String, double, double)} for matching a 1D range.
*
- {@link #newSetQuery(String, double...)} for matching a 1D set.
*
*
* @see PointValues
*/
public final class DoubleField extends Field {
private static final FieldType FIELD_TYPE = new FieldType();
private static final FieldType FIELD_TYPE_STORED;
static {
FIELD_TYPE.setDimensions(1, Double.BYTES);
FIELD_TYPE.setDocValuesType(DocValuesType.SORTED_NUMERIC);
FIELD_TYPE.freeze();
FIELD_TYPE_STORED = new FieldType(FIELD_TYPE);
FIELD_TYPE_STORED.setStored(true);
FIELD_TYPE_STORED.freeze();
}
private final StoredValue storedValue;
/**
* Creates a new DoubleField, indexing the provided point, storing it as a DocValue, and
* optionally storing it as a stored field.
*
* @param name field name
* @param value the double value
* @param stored whether to store the field
* @throws IllegalArgumentException if the field name or value is null.
*/
public DoubleField(String name, double value, Field.Store stored) {
super(name, stored == Field.Store.YES ? FIELD_TYPE_STORED : FIELD_TYPE);
fieldsData = NumericUtils.doubleToSortableLong(value);
if (stored == Field.Store.YES) {
storedValue = new StoredValue(value);
} else {
storedValue = null;
}
}
@Override
public BytesRef binaryValue() {
byte[] encodedPoint = new byte[Double.BYTES];
double value = getValueAsDouble();
DoublePoint.encodeDimension(value, encodedPoint, 0);
return new BytesRef(encodedPoint);
}
private double getValueAsDouble() {
return NumericUtils.sortableLongToDouble(numericValue().longValue());
}
@Override
public StoredValue storedValue() {
return storedValue;
}
@Override
public String toString() {
return getClass().getSimpleName() + " <" + name + ':' + getValueAsDouble() + '>';
}
@Override
public void setDoubleValue(double value) {
super.setLongValue(NumericUtils.doubleToSortableLong(value));
if (storedValue != null) {
storedValue.setDoubleValue(value);
}
}
@Override
public void setLongValue(long value) {
throw new IllegalArgumentException("cannot change value type from Double to Long");
}
/**
* Create a query for matching an exact double value.
*
* @param field field name. must not be {@code null}.
* @param value exact value
* @throws IllegalArgumentException if {@code field} is null.
* @return a query matching documents with this exact value
*/
public static Query newExactQuery(String field, double value) {
return newRangeQuery(field, value, value);
}
/**
* Create a range query for double values.
*
* You can have half-open ranges (which are in fact </≤ or >/≥ queries) by setting
* {@code lowerValue = Double.NEGATIVE_INFINITY} or {@code upperValue = Double.POSITIVE_INFINITY}.
*
*
Range comparisons are consistent with {@link Double#compareTo(Double)}.
*
* @param field field name. must not be {@code null}.
* @param lowerValue lower portion of the range (inclusive).
* @param upperValue upper portion of the range (inclusive).
* @throws IllegalArgumentException if {@code field} is null.
* @return a query matching documents within this range.
*/
public static Query newRangeQuery(String field, double lowerValue, double upperValue) {
PointRangeQuery.checkArgs(field, lowerValue, upperValue);
return new IndexOrDocValuesQuery(
DoublePoint.newRangeQuery(field, lowerValue, upperValue),
SortedNumericDocValuesField.newSlowRangeQuery(
field,
NumericUtils.doubleToSortableLong(lowerValue),
NumericUtils.doubleToSortableLong(upperValue)));
}
/**
* Create a query matching values in a supplied set
*
* @param field field name. must not be {@code null}.
* @param values double values
* @throws IllegalArgumentException if {@code field} is null.
* @return a query matching documents within this set.
*/
public static Query newSetQuery(String field, double... values) {
if (field == null) {
throw new IllegalArgumentException("field cannot be null");
}
long points[] = new long[values.length];
for (int i = 0; i < values.length; i++) {
points[i] = NumericUtils.doubleToSortableLong(values[i]);
}
return new IndexOrDocValuesQuery(
DoublePoint.newSetQuery(field, values.clone()),
SortedNumericDocValuesField.newSlowSetQuery(field, points));
}
/**
* Create a new {@link SortField} for double values.
*
* @param field field name. must not be {@code null}.
* @param reverse true if natural order should be reversed.
* @param selector custom selector type for choosing the sort value from the set.
*/
public static SortField newSortField(
String field, boolean reverse, SortedNumericSelector.Type selector) {
return new SortedNumericSortField(field, SortField.Type.DOUBLE, reverse, selector);
}
}