2

我正在使用 BaseX XQJ API 在我的 java 应用程序中对 XML 文件执行 xquery 查询。我构建的以下 xquery 很好地生成了我想要的输出:

let $doc := doc("eprints")
for $i in distinct-values($doc//issn)
let $jn := $doc//paper[issn = $i]/publication
where (count(distinct-values($jn)) > 1)
return <issn num="{$i}">"{$jn}"</issn>

在通过在 BaseX 应用程序中测试这个查询来说服自己这个查询有效之后,我在 java 代码中实现了这个查询。

public static void main(String[] args) throws XQException{
    XQDataSource ds = new BaseXXQDataSource();
    ds.setProperty("serverName", "localhost");
    ds.setProperty("port", "1984");
    ds.setProperty("user", "xxxx");
    ds.setProperty("password", "xxxxx");
    ds.setProperty("databaseName", "eprints");

    XQConnection conn = ds.getConnection("admin", "admin");

    XQExpression xqe = conn.createExpression();
    XQResultSequence result = xqe.executeQuery("let $doc := doc(\"eprints\")"+
                                            "for $i in distinct-values($doc//issn)"+
                                            "let $jn := $doc//paper[issn = $i]/publication"+
                                            "where (count(distinct-values($jn)) > 1)"+
                                            "return <issn num='{$i}'>'{jn}'</issn>"

                                            ); 
}

但是,此代码导致错误,指出找不到 $jn 变量:线程“main”中的异常 javax.xml.xquery.XQQueryException:[XPST0008]:未定义的变量 $jn。查看查询时,可以看到 $jn 实际上是在 for 语句的 let 语句中定义的。

但是,当我在使用 $jn 的位置直接插入分配给 $jn 的表达式时,代码确实可以正常工作:

public static void main(String[] args) throws XQException{
    XQDataSource ds = new BaseXXQDataSource();
    ds.setProperty("serverName", "localhost");
    ds.setProperty("port", "1984");
    ds.setProperty("user", "admin");
    ds.setProperty("password", "admin");
    ds.setProperty("databaseName", "eprints");

    XQConnection conn = ds.getConnection("admin", "admin");

    XQExpression xqe = conn.createExpression();
    XQResultSequence result = xqe.executeQuery("let $doc := doc(\"eprints\")"+
                                            "for $i in distinct-values($doc//issn)"+
                                            "where (count(distinct-values($doc//paper[issn = $i]/publication)) > 1)"+
                                            "return <issn num='{$i}'>'{$doc//paper[issn = $i]/publication}'</issn>"
                                            );    
}

BaseX XQJ API 似乎无法处理在 for 语句中有 let 语句的查询。有谁知道错误的原因是什么?

4

1 回答 1

3

publication原始查询之间和中没有空格where(由字符串连接伪装),因此这两行都被解释为轴路径和动态函数应用程序:

let $jn := $doc//paper[issn = $i]/publicationwhere (count(distinct-values($jn)) > 1)

这会产生错误,因为不允许递归变量定义。只需在 之后插入一个空格publication,它应该可以正常运行。

于 2012-05-10T11:23:30.890 回答