Saxonica.com

Calling XQuery Functions from Java

Although the usual way to invoke XQuery from a Java application is to compile and execute a query as described above, it is also possible to invoke individual XQuery functions directly from Java if required. This interface is very efficient but performs less validation of parameters than the standard interface described above. To achieve this, first compile the query as described above, specifying a query that includes one or more declare function declarations in the query prolog. It is then possible to retrieve the UserFunction objects representing the compiled code of these functions, by calling the getUserDefinedFunction method on the StaticQueryContext object. As discussed above, the information is not held in the original StaticQueryContext object (which Saxon does not alter), but in a modified copy which is obtainable from the XQueryExpression object. Once found, such a function can be called using its call method. The first argument to this is an array containing the values of the arguments. These must be supplied using Saxon's native classes, for example an integer argument is supplied as an instance of net.sf.saxon.value.IntegerValue. These values must be of the type expected by the function; no conversion or type checking takes place (which means that a ClassCastException will probably occur if the wrong type of value is supplied). The second argument to the call method is a Controller. A Controller can be obtained by calling the getController() method on the XQueryExpression object. The same Controller can be used for a series of separate function calls.

For example:


StaticQueryContext sqc = 
        new StaticQueryContext(new Configuration());

XQueryExpression exp1 = sqc.compileQuery(
        "declare namespace f='f.ns';" +
        "declare function f:t1($p as xs:integer) { $p * $p };" +
        "declare function f:t2($p as xs:integer) { $p + $p };" +
        "1"
);

StaticQueryContext sqc2 = exp1.getStaticContext();
UserFunction fn1 = sqc2.getUserDefinedFunction("f.ns", "t1", 1);
UserFunction fn2 = sqc2.getUserDefinedFunction("f.ns", "t2", 1);
Controller controller = exp1.getController();

IntegerValue[] arglist = new IntegerValue[1];
for (int x=1; x<1000000; x++) {
    arglist[0] = new IntegerValue(x);
    Value v1 = fn1.call(arglist, controller);
    Value v2 = fn2.call(arglist, controller);
    System.err.println("Returned product " + v1 + "; sum =" + v2);
}

Next