/*
* 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.commons.jxpath.ri.compiler;
import org.apache.commons.jxpath.NodeSet;
import org.apache.commons.jxpath.Pointer;
import org.apache.commons.jxpath.ri.EvalContext;
import org.apache.commons.jxpath.ri.model.NodePointer;
import org.apache.commons.jxpath.ri.QName;
import org.apache.commons.jxpath.util.ValueUtils;
import java.util.Collections;
import java.util.Iterator;
import java.util.Locale;
/**
* Common superclass for several types of nodes in the parse tree. Provides
* APIs for optimization of evaluation of expressions. Specifically, an
* expression only needs to executed once during the evaluation of an xpath
* if that expression is context-independent. Expression.isContextDependent()
* provides that hint.
*
* @author Dmitri Plotnikov
* @version $Revision: 652845 $ $Date: 2008-05-02 12:46:46 -0500 (Fri, 02 May 2008) $
*/
public abstract class Expression {
/** zero */
protected static final Double ZERO = new Double(0);
/** one */
protected static final Double ONE = new Double(1);
/** NaN */
protected static final Double NOT_A_NUMBER = new Double(Double.NaN);
private boolean contextDependencyKnown = false;
private boolean contextDependent;
/**
* Returns true if this expression should be re-evaluated
* each time the current position in the context changes.
* @return boolean
*/
public synchronized boolean isContextDependent() {
if (!contextDependencyKnown) {
contextDependent = computeContextDependent();
contextDependencyKnown = true;
}
return contextDependent;
}
/**
* Implemented by subclasses and result is cached by isContextDependent()
* @return calculated context-dependentness as boolean
*/
public abstract boolean computeContextDependent();
/**
* Evaluates the expression. If the result is a node set, returns
* the first element of the node set.
* @param context evaluation context
* @return Object
*/
public abstract Object computeValue(EvalContext context);
/**
* Evaluates the expression. If the result is a node set, returns
* the first element of the node set.
* @param context evaluation context
* @return Object
*/
public abstract Object compute(EvalContext context);
/**
* Iterate over the values from the specified context.
* @param context evaluation context
* @return value Iterator
*/
public Iterator iterate(EvalContext context) {
Object result = compute(context);
if (result instanceof EvalContext) {
return new ValueIterator((EvalContext) result);
}
if (result instanceof NodeSet) {
return new ValueIterator(((NodeSet) result).getPointers().iterator());
}
return ValueUtils.iterate(result);
}
/**
* Iterate over the pointers from the specified context.
* @param context evaluation context
* @return pointer Iterator
*/
public Iterator iteratePointers(EvalContext context) {
Object result = compute(context);
if (result == null) {
return Collections.EMPTY_LIST.iterator();
}
if (result instanceof EvalContext) {
return (EvalContext) result;
}
if (result instanceof NodeSet) {
return new PointerIterator(((NodeSet) result).getPointers().iterator(),
new QName(null, "value"),
context.getRootContext().getCurrentNodePointer().getLocale());
}
return new PointerIterator(ValueUtils.iterate(result),
new QName(null, "value"),
context.getRootContext().getCurrentNodePointer().getLocale());
}
/**
* Pointer iterator
*/
public static class PointerIterator implements Iterator {
private Iterator iterator;
private QName qname;
private Locale locale;
//to what method does the following comment refer?
/**
* Create a new PointerIterator
* @param it underlying Iterator
* @param qname name
* @param locale Locale
* @deprecated Use the method that takes a NamespaceManager
*/
public PointerIterator(Iterator it, QName qname, Locale locale) {
this.iterator = it;
this.qname = qname;
this.locale = locale;
}
public boolean hasNext() {
return iterator.hasNext();
}
public Object next() {
Object o = iterator.next();
return o instanceof Pointer ? o : NodePointer.newNodePointer(qname, o, locale);
}
/**
* Unsupported.
*/
public void remove() {
throw new UnsupportedOperationException();
}
}
/**
* Value Iterator
*/
public static class ValueIterator implements Iterator {
private Iterator iterator;
/**
* Create a new ValueIterator.
* @param it underlying Iterator, may contain pointers
*/
public ValueIterator(Iterator it) {
this.iterator = it;
}
public boolean hasNext() {
return iterator.hasNext();
}
public Object next() {
Object o = iterator.next();
return o instanceof Pointer ? ((Pointer) o).getValue() : o;
}
/**
* Unsupported.
*/
public void remove() {
throw new UnsupportedOperationException();
}
}
}
|