keycall.java

来自「JAVA的一些源码 JAVA2 STANDARD EDITION DEVELO」· Java 代码 · 共 370 行

JAVA
370
字号
/* * Copyright 2001-2004 The Apache Software Foundation. * * Licensed 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. *//* * $Id: KeyCall.java,v 1.15 2004/02/16 22:24:29 minchau Exp $ */package com.sun.org.apache.xalan.internal.xsltc.compiler;import java.util.Vector;import com.sun.org.apache.bcel.internal.generic.ALOAD;import com.sun.org.apache.bcel.internal.generic.ASTORE;import com.sun.org.apache.bcel.internal.generic.BranchHandle;import com.sun.org.apache.bcel.internal.generic.ConstantPoolGen;import com.sun.org.apache.bcel.internal.generic.GOTO;import com.sun.org.apache.bcel.internal.generic.IFGT;import com.sun.org.apache.bcel.internal.generic.INVOKEINTERFACE;import com.sun.org.apache.bcel.internal.generic.INVOKESPECIAL;import com.sun.org.apache.bcel.internal.generic.INVOKEVIRTUAL;import com.sun.org.apache.bcel.internal.generic.InstructionHandle;import com.sun.org.apache.bcel.internal.generic.InstructionList;import com.sun.org.apache.bcel.internal.generic.LocalVariableGen;import com.sun.org.apache.bcel.internal.generic.NEW;import com.sun.org.apache.bcel.internal.generic.PUSH;import com.sun.org.apache.xalan.internal.xsltc.compiler.util.ClassGenerator;import com.sun.org.apache.xalan.internal.xsltc.compiler.util.MethodGenerator;import com.sun.org.apache.xalan.internal.xsltc.compiler.util.StringType;import com.sun.org.apache.xalan.internal.xsltc.compiler.util.Type;import com.sun.org.apache.xalan.internal.xsltc.compiler.util.TypeCheckError;import com.sun.org.apache.xalan.internal.xsltc.compiler.util.Util;/** * @author Morten Jorgensen * @author Santiago Pericas-Geertsen */final class KeyCall extends FunctionCall {    /**     * The name of the key.     */    private Expression _name;    /**     * The value to look up in the key/index.     */    private Expression _value;    /**     * The value's data type.     */    private Type _valueType; // The value's data type    /**     * Expanded qname when name is literal.     */    private QName _resolvedQName = null;    /**     * Get the parameters passed to function:     *   key(String name, String value)     *   key(String name, NodeSet value)     * The 'arguments' vector should contain two parameters for key() calls,     * one holding the key name and one holding the value(s) to look up. The     * vector has only one parameter for id() calls (the key name is always     * "##id" for id() calls).     *     * @param fname The function name (should be 'key' or 'id')     * @param arguments A vector containing the arguments the the function     */    public KeyCall(QName fname, Vector arguments) {	super(fname, arguments);	switch(argumentCount()) {	case 1:	    _name = null;	    _value = argument(0);	    break;	case 2:	    _name = argument(0);	    _value = argument(1);	    break;	default:	    _name = _value = null;	    break;	}    }    /**     * If this call to key() is in a top-level element like another variable     * or param, add a dependency between that top-level element and the     * referenced key. For example,     *     * <xsl:key name="x" .../>     * <xsl:variable name="y" select="key('x', 1)"/>     *     * and assuming this class represents "key('x', 1)", add a reference     * between variable y and key x. Note that if 'x' is unknown statically     * in key('x', 1), there's nothing we can do at this point.     */     public void addParentDependency() {        // If name unknown statically, there's nothing we can do        if (_resolvedQName == null) return; 	SyntaxTreeNode node = this; 	while (node != null && node instanceof TopLevelElement == false) { 	    node = node.getParent(); 	} 		 	TopLevelElement parent = (TopLevelElement) node; 	if (parent != null) {            parent.addDependency(getSymbolTable().getKey(_resolvedQName)); 	}    }    /**     * Type check the parameters for the id() or key() function.     * The index name (for key() call only) must be a string or convertable     * to a string, and the lookup-value must be a string or a node-set.     * @param stable The parser's symbol table     * @throws TypeCheckError When the parameters have illegal type     */    public Type typeCheck(SymbolTable stable) throws TypeCheckError {	final Type returnType = super.typeCheck(stable);	// Run type check on the key name (first argument) - must be a string,	// and if it is not it must be converted to one using string() rules.	if (_name != null) {	    final Type nameType = _name.typeCheck(stable); 	    if (_name instanceof LiteralExpr) {		final LiteralExpr literal = (LiteralExpr) _name;		_resolvedQName = 		    getParser().getQNameIgnoreDefaultNs(literal.getValue());	    }	    else if (nameType instanceof StringType == false) {		_name = new CastExpr(_name, Type.String);	    }	}	// Run type check on the value for this key. This value can be of	// any data type, so this should never cause any type-check errors.	// If the value is not a node-set then it should be converted to a	// string before the lookup is done. If the value is a node-set then	// this process (convert to string, then do lookup) should be applied	// to every node in the set, and the result from all lookups should	// be added to the resulting node-set.	_valueType = _value.typeCheck(stable);	if (_valueType != Type.NodeSet && _valueType != Type.String) 	{	    _value = new CastExpr(_value, Type.String);	}        // If in a top-level element, create dependency to the referenced key        addParentDependency();	return returnType;    }    /**     * This method is called when the constructor is compiled in     * Stylesheet.compileConstructor() and not as the syntax tree is traversed.     * This method is a wrapper for the real translation method, which is     * the private method translateCall() below. All this method does is to     * wrap the KeyIndex that this function returns inside a duplicate filter.     * The duplicate filter is used both to eliminate duplicates and to     * cache the nodes in the index.     * @param classGen The Java class generator     * @param methodGen The method generator     */    public void translate(ClassGenerator classGen,			  MethodGenerator methodGen) {	final ConstantPoolGen cpg = classGen.getConstantPool();	final InstructionList il = methodGen.getInstructionList();						 	final int getNodeHandle = cpg.addInterfaceMethodref(DOM_INTF,							   "getNodeHandle",							   "(I)"+NODE_SIG);		// Wrap the KeyIndex (iterator) inside a duplicate filter iterator	// to pre-read the indexed nodes and cache them.	final int dupInit = cpg.addMethodref(DUP_FILTERED_ITERATOR,					     "<init>",					     "("+NODE_ITERATOR_SIG+")V");					     	il.append(new NEW(cpg.addClass(DUP_FILTERED_ITERATOR)));		il.append(DUP);	translateCall(classGen, methodGen);	il.append(new INVOKESPECIAL(dupInit));	    }    /**     * Translate the actual index lookup - leaves KeyIndex (iterator) on stack     * @param classGen The Java class generator     * @param methodGen The method generator     */    private void translateCall(ClassGenerator classGen,			      MethodGenerator methodGen) {	final ConstantPoolGen cpg = classGen.getConstantPool();	final InstructionList il = methodGen.getInstructionList();	// Returns the string value for a node in the DOM	final int getNodeValue = cpg.addInterfaceMethodref(DOM_INTF,							   GET_NODE_VALUE,							   "(I)"+STRING_SIG);	// Returns the KeyIndex object of a given name	final int getKeyIndex = cpg.addMethodref(TRANSLET_CLASS,						 "getKeyIndex",						 "(Ljava/lang/String;)"+						 KEY_INDEX_SIG);	// Initialises a KeyIndex to return nodes with specific values	final int lookupId = cpg.addMethodref(KEY_INDEX_CLASS,					      "lookupId",					      "(Ljava/lang/Object;)V");	final int lookupKey = cpg.addMethodref(KEY_INDEX_CLASS,					       "lookupKey",					       "(Ljava/lang/Object;)V");	// Merges the nodes in two KeyIndex objects	final int merge = cpg.addMethodref(KEY_INDEX_CLASS,					   "merge",					   "("+KEY_INDEX_SIG+")V");	// Constructor for KeyIndex class	final int indexConstructor = cpg.addMethodref(TRANSLET_CLASS,						      "createKeyIndex",						      "()"+KEY_INDEX_SIG);						      	// KeyIndex.setDom(Dom) => void	final int keyDom = cpg.addMethodref(XSLT_PACKAGE + ".dom.KeyIndex",					 "setDom",					 "("+DOM_INTF_SIG+")V");				 						      		// This local variable holds the index/iterator we will return	final LocalVariableGen returnIndex =	    methodGen.addLocalVariable("returnIndex",				       Util.getJCRefType(KEY_INDEX_SIG),				       il.getEnd(), null);	// This local variable holds the index we're using for search	final LocalVariableGen searchIndex =	    methodGen.addLocalVariable("searchIndex",				       Util.getJCRefType(KEY_INDEX_SIG),				       il.getEnd(), null);	// If the second paramter is a node-set we need to go through each	// node in the set, convert each one to a string and do a look up in	// the named index, and then merge all the resulting node sets.	if (_valueType == Type.NodeSet) {	    // Save current node and current iterator on the stack	    il.append(methodGen.loadCurrentNode());	    il.append(methodGen.loadIterator());	    // Get new iterator from 2nd parameter node-set & store in variable	    _value.translate(classGen, methodGen);	    _value.startIterator(classGen, methodGen);	    il.append(methodGen.storeIterator());	    // Create the KeyIndex object (the iterator) we'll return	    il.append(classGen.loadTranslet());	    il.append(new INVOKEVIRTUAL(indexConstructor));	    il.append(DUP);	    il.append(methodGen.loadDOM());	    il.append(new INVOKEVIRTUAL(keyDom));	    il.append(new ASTORE(returnIndex.getIndex()));	    // Initialise the index specified in the first parameter of key()	    il.append(classGen.loadTranslet());	    if (_name == null) {		il.append(new PUSH(cpg,"##id"));	    }	    else if (_resolvedQName != null) {		il.append(new PUSH(cpg, _resolvedQName.toString()));	    }	    else {		_name.translate(classGen, methodGen);	    }	    il.append(new INVOKEVIRTUAL(getKeyIndex));	    il.append(new ASTORE(searchIndex.getIndex()));	    // LOOP STARTS HERE	    // Now we're ready to start traversing the node-set given in	    // the key() function's second argument....	    final BranchHandle nextNode = il.append(new GOTO(null));	    final InstructionHandle loop = il.append(NOP);	    // Push returnIndex on stack to prepare for call to merge()	    il.append(new ALOAD(returnIndex.getIndex()));	    	    // Lookup index using the string value from the current node	    il.append(new ALOAD(searchIndex.getIndex()));	    il.append(DUP);	    il.append(methodGen.loadDOM());	    il.append(methodGen.loadCurrentNode());	    il.append(new INVOKEINTERFACE(getNodeValue, 2));	    if (_name == null) {		il.append(new INVOKEVIRTUAL(lookupId));	    }	    else {		il.append(new INVOKEVIRTUAL(lookupKey));	    }	    // Call to returnIndex.merge(searchIndex);	    il.append(new INVOKEVIRTUAL(merge));			    // Go on with next node in the 2nd parameter node-set	    nextNode.setTarget(il.append(methodGen.loadIterator()));	    il.append(methodGen.nextNode());	    il.append(DUP);	    il.append(methodGen.storeCurrentNode());	    il.append(new IFGT(loop));	    // LOOP ENDS HERE	    // Restore current node and current iterator from the stack	    il.append(methodGen.storeIterator());	    il.append(methodGen.storeCurrentNode());	    // Return with the an iterator for all resulting nodes	    il.append(new ALOAD(returnIndex.getIndex()));	}	// If the second parameter is a single value we just lookup the named	// index and initialise the iterator to return nodes with this value.	else {	    // Call getKeyIndex in AbstractTranslet with the name of the key	    // to get the index for this key (which is also a node iterator).	    il.append(classGen.loadTranslet());	    if (_name == null) {		il.append(new PUSH(cpg,"##id"));	    }	    else if (_resolvedQName != null) {		il.append(new PUSH(cpg, _resolvedQName.toString()));	    }	    else {		_name.translate(classGen, methodGen);	    }	    il.append(new INVOKEVIRTUAL(getKeyIndex));	    // Now use the value in the second argument to determine what nodes	    // the iterator should return.	    il.append(DUP);	    _value.translate(classGen, methodGen);	    if (_name == null) {		il.append(new INVOKEVIRTUAL(lookupId));	    }	    else {		il.append(new INVOKEVIRTUAL(lookupKey));	    }	}    }}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?