net.sf.saxon.value.SingletonClosure Maven / Gradle / Ivy
package net.sf.saxon.value;
import net.sf.saxon.event.SequenceReceiver;
import net.sf.saxon.expr.XPathContext;
import net.sf.saxon.expr.Expression;
import net.sf.saxon.om.Item;
import net.sf.saxon.om.NodeInfo;
import net.sf.saxon.om.SequenceIterator;
import net.sf.saxon.om.SingletonIterator;
import net.sf.saxon.trans.XPathException;
/**
* A SingletonClosure represents a value that has not yet been evaluated: the value is represented
* by an expression, together with saved values of all the context variables that the
* expression depends on. The value of a SingletonClosure is always either a single item
* or an empty sequence.
*
* The expression may depend on local variables and on the context item; these values
* are held in the saved XPathContext object that is kept as part of the Closure, and they
* will always be read from that object. The expression may also depend on global variables;
* these are unchanging, so they can be read from the Bindery in the normal way. Expressions
* that depend on other contextual information, for example the values of position(), last(),
* current(), current-group(), should not be evaluated using this mechanism: they should
* always be evaluated eagerly. This means that the Closure does not need to keep a copy
* of these context variables.
*/
public class SingletonClosure extends Closure {
private boolean built = false;
private Item value = null;
/**
* Constructor should not be called directly, instances should be made using the make() method.
* @param exp the expression to be lazily evaluated
* @param context the context in which the expression should be evaluated
*/
public SingletonClosure(Expression exp, XPathContext context) throws XPathException {
expression = exp;
savedXPathContext = context.newContext();
saveContext(exp, context);
//System.err.println("Creating SingletonClosure");
}
/**
* Evaluate the expression in a given context to return an iterator over a sequence
*/
public SequenceIterator iterate() throws XPathException {
return SingletonIterator.makeIterator(asItem());
}
/**
* Process the expression by writing the value to the current Receiver
*
* @param context The dynamic context, giving access to the current node,
* the current variables, etc.
*/
public void process(XPathContext context) throws XPathException {
SequenceReceiver out = context.getReceiver();
out.append(asItem(), 0, NodeInfo.ALL_NAMESPACES);
}
/**
* Return the value in the form of an Item
*
* @return the value in the form of an Item
*/
public Item asItem() throws XPathException {
if (!built) {
value = expression.evaluateItem(savedXPathContext);
built = true;
savedXPathContext = null; // release variables saved in the context to the garbage collector
}
return value;
}
/**
* Get the n'th item in the sequence (starting from 0). This is defined for all
* SequenceValues, but its real benefits come for a SequenceValue stored extensionally
*/
public Item itemAt(int n) throws XPathException {
if (n != 0) {
return null;
}
return asItem();
}
/**
* Get the length of the sequence
*/
public int getLength() throws XPathException {
return (asItem() == null ? 0 : 1);
}
/**
* Return a value containing all the items in the sequence returned by this
* SequenceIterator
*
* @return the corresponding value
*/
public Value materialize() throws XPathException {
return Value.asValue(asItem());
}
}
//
// 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.
//
// Contributor(s):
//
© 2015 - 2025 Weber Informatics LLC | Privacy Policy