8

假设我有一些耶拿查询对象:

String query = "SELECT * WHERE{ ?s <some_uri> ?o ...etc. }";
Query q = QueryFactory.create(query, Syntax.syntaxARQ);

在查询中获取三元组的所有主题的最佳方法是什么?最好不必手动进行任何字符串解析/操作。

例如,给定一个查询

SELECT * WHERE {
    ?s ?p ?o;
       ?p2 ?o2.
    ?s2 ?p3 ?o3.
    ?s3 ?p4 ?o4.
    <http://example.com> ?p5 ?o5.
}

我希望返回一些看起来像的列表

[?s, ?s2, ?s3, <http://example.com>]

换句话说,我想要查询中所有主题的列表。即使只有那些是变量的主题或那些是文字/uris的主题也会很有用,但我想在查询中找到所有主题的列表。

我知道有一些方法可以返回结果变量 ( Query.getResultVars) 和一些其他信息(请参阅http://jena.apache.org/documentation/javadoc/arq/com/hp/hpl/jena/query/Query.html),但是我似乎找不到任何可以专门获取查询主题的内容(所有结果变量的列表也将返回谓词和对象)。

任何帮助表示赞赏。

4

2 回答 2

11

有趣的问题。您需要做的是通过查询,并为每个三元组块迭代并查看第一部分。

最可靠的方法是通过一个元素遍历器,它将遍历查询的每个部分。在您的情况下,它可能看起来很重要,但查询可以包含各种各样的东西,包括FILTERs,OPTIONALs和 nested SELECTs。使用 walker 意味着你可以忽略那些东西,只专注于你想要的东西:

Query q = QueryFactory.create(query); // SPARQL 1.1

// Remember distinct subjects in this
final Set<Node> subjects = new HashSet<Node>();

// This will walk through all parts of the query
ElementWalker.walk(q.getQueryPattern(),
    // For each element...
    new ElementVisitorBase() {
        // ...when it's a block of triples...
        public void visit(ElementPathBlock el) {
            // ...go through all the triples...
            Iterator<TriplePath> triples = el.patternElts();
            while (triples.hasNext()) {
                // ...and grab the subject
                subjects.add(triples.next().getSubject());
            }
        }
    }
);
于 2013-03-04T16:13:29.483 回答
1

可能为时已晚,但另一种方法是利用 Jena ARQ 库并创建给定查询的代数。一旦创建了代数,就可以对其进行编译,并且可以遍历所有三元组(在 where 子句中给出)。下面是代码,希望对你有帮助:

Query query = qExec.getQuery(); //qExec is an object of QueryExecutionFactory

// Generate algebra of the query
Op op = Algebra.compile(query);
CustomOpVisitorBase opVisitorBase = new CustomOpVisitorBase();
opVisitorBase.opVisitorWalker(op);
List<Triple> queryTriples = opVisitorBase.triples;

CustomOpVisitor 类如下:

public class CustomOpVisitorBase extends OpVisitorBase {
List<Triple> triples = null;
void opVisitorWalker(Op op) {
    OpWalker.walk(op, this);
}

@Override
public void visit(final OpBGP opBGP) {
    triples = opBGP.getPattern().getList();
}
}

遍历 Triples 列表并使用给定的属性函数,例如triple.getSubject() 等。

于 2020-02-06T11:10:20.817 回答