net.sf.saxon.expr.RangeExpression Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of saxon9 Show documentation
Show all versions of saxon9 Show documentation
Provides a basic XSLT 2.0 and XQuery 1.0 processor (W3C Recommendations,
January 2007). Command line interfaces and implementations of several
Java APIs (DOM, XPath, s9api) are also included.
The newest version!
package net.sf.saxon.expr;
import net.sf.saxon.om.EmptyIterator;
import net.sf.saxon.om.SequenceIterator;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.type.BuiltInAtomicType;
import net.sf.saxon.type.ItemType;
import net.sf.saxon.type.TypeHierarchy;
import net.sf.saxon.value.*;
/**
* A RangeExpression is an expression that represents an integer sequence as
* a pair of end-points (for example "x to y").
* If the end-points are equal, the sequence is of length one.
* From Saxon 7.8, the sequence must be ascending; if the end-point is less
* than the start-point, an empty sequence is returned. This is to allow
* expressions of the form "for $i in 1 to count($seq) return ...."
*/
public class RangeExpression extends BinaryExpression {
/**
* Construct a RangeExpression
* @param start expression that computes the start of the range
* @param op represents the operator "to", needed only because this class is a subclass of
* BinaryExpression which needs an operator
* @param end expression that computes the end of the range
*/
public RangeExpression(Expression start, int op, Expression end) {
super(start, op, end);
}
/**
* Type-check the expression
*/
public Expression typeCheck(ExpressionVisitor visitor, ItemType contextItemType) throws XPathException {
operand0 = visitor.typeCheck(operand0, contextItemType);
operand1 = visitor.typeCheck(operand1, contextItemType);
RoleLocator role0 = new RoleLocator(RoleLocator.BINARY_EXPR, "to", 0, null);
role0.setSourceLocator(this);
operand0 = TypeChecker.staticTypeCheck(
operand0, SequenceType.OPTIONAL_INTEGER, false, role0, visitor);
RoleLocator role1 = new RoleLocator(RoleLocator.BINARY_EXPR, "to", 1, null);
role1.setSourceLocator(this);
operand1 = TypeChecker.staticTypeCheck(
operand1, SequenceType.OPTIONAL_INTEGER, false, role1, visitor);
return makeConstantRange();
}
/**
* Perform optimisation of an expression and its subexpressions.
*
* This method is called after all references to functions and variables have been resolved
* to the declaration of the function or variable, and after all type checking has been done.
*
* @param visitor an expression visitor
* @param contextItemType the static type of "." at the point where this expression is invoked.
* The parameter is set to null if it is known statically that the context item will be undefined.
* If the type of the context item is not known statically, the argument is set to
* {@link net.sf.saxon.type.Type#ITEM_TYPE}
* @return the original expression, rewritten if appropriate to optimize execution
* @throws XPathException if an error is discovered during this phase
* (typically a type error)
*/
public Expression optimize(ExpressionVisitor visitor, ItemType contextItemType) throws XPathException {
operand0 = visitor.optimize(operand0, contextItemType);
operand1 = visitor.optimize(operand1, contextItemType);
return makeConstantRange();
}
private Expression makeConstantRange() {
if (operand0 instanceof Literal && operand1 instanceof Literal) {
Value v0 = ((Literal)operand0).getValue();
Value v1 = ((Literal)operand1).getValue();
if (v0 instanceof Int64Value && v1 instanceof Int64Value) {
long i0 = ((Int64Value)v0).longValue();
long i1 = ((Int64Value)v1).longValue();
Literal result;
if (i0 > i1) {
result = Literal.makeEmptySequence();
} else if (i0 == i1) {
result = Literal.makeLiteral(Int64Value.makeIntegerValue(i0));
} else {
result = Literal.makeLiteral(new IntegerRange(i0, i1));
}
ExpressionTool.copyLocationInfo(this, result);
return result;
}
}
return this;
}
/**
* Get the data type of the items returned
* @param th the type hierarchy cache
*/
public ItemType getItemType(TypeHierarchy th) {
return BuiltInAtomicType.INTEGER;
}
/**
* Determine the static cardinality
*/
public int computeCardinality() {
return StaticProperty.ALLOWS_ZERO_OR_MORE;
}
/**
* Copy an expression. This makes a deep copy.
*
* @return the copy of the original expression
*/
public Expression copy() {
return new RangeExpression(operand0.copy(), operator, operand1.copy());
}
/**
* Return an iteration over the sequence
*/
public SequenceIterator iterate(XPathContext context) throws XPathException {
AtomicValue av1 = (AtomicValue)operand0.evaluateItem(context);
if (av1 == null) {
return EmptyIterator.getInstance();
}
NumericValue v1 = (NumericValue)av1;
AtomicValue av2 = (AtomicValue)operand1.evaluateItem(context);
if (av2 == null) {
return EmptyIterator.getInstance();
}
NumericValue v2 = (NumericValue)av2;
if (v1.compareTo(v2) > 0) {
return EmptyIterator.getInstance();
}
return new RangeIterator(v1.longValue(), v2.longValue());
}
}
//
// The contents of this file are subject to the Mozilla Public License Version 1.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.mozilla.org/MPL/
//
// Software distributed under the License is distributed on an "AS IS" basis,
// WITHOUT WARRANTY OF ANY KIND, either express or implied.
// See the License for the specific language governing rights and limitations under the License.
//
// The Original Code is: all this file.
//
// The Initial Developer of the Original Code is Michael H. Kay
//
// Portions created by (your name) are Copyright (C) (your legal entity). All Rights Reserved.
//
// Contributor(s): none.
//
© 2015 - 2025 Weber Informatics LLC | Privacy Policy