【问题标题】:RDF list subjects with their objects in a single lineRDF 在一行中列出主题及其对象
【发布时间】:2013-08-28 10:32:12
【问题描述】:

我有一个 RDF 文件,我需要从中提取一些信息并将其写入文件。我理解它基本上是如何工作的,但我坚持这一点:

String queryString = "select ?person ?children where { ?person ?hasChildren ?children}";
TupleQuery tupleQuery = conn.prepareTupleQuery(QueryLanguage.SPARQL, queryString);
TupleQueryResult result = tupleQuery.evaluate();     
while (result.hasNext()) {
    BindingSet bindingSet = result.next();
    Value p1 = bindingSet.getValue("person");
    Value p2 = bindingSet.getValue("child");
    println(p1 + " has children " + p2 +"");
}
result.close();

我得到的输出是这样的:

http://example.org/people/person1 has children http://example.org/people/child1
http://example.org/people/person1 has children http://example.org/people/child2

我不知道如何以这种格式列出所有人员及其对象:

person1 has children child1 and child2

如何做到这一点?

【问题讨论】:

  • 此代码不可能是您实际运行的代码;该查询绑定了变量?children,但您正在检索名为?child 的变量的值。

标签: rdf sparql jena sesame


【解决方案1】:

您可能会发现这个描述 SPARQL 的 group_concat 的答案很有用:

在 SPARQL 中,当您有一组查询解决方案的结果集时,您可以在一个或多个变量上group,合并具有这些共同变量的解决方案。例如,考虑数据

@prefix : <http://example.org/people/>.

:person1 :hasChild :child1, :child2, :child3 .
:person2 :hasChild :child4, :child5 .
:person3 :hasChild :child6 .

如果您对其运行以下查询

prefix : <http://example.org/people/>

select ?person ?child where { 
  ?person :hasChild ?child .
}

你会得到这样的结果:

$ arq --data data.n3 --query query.sparql
----------------------
| person   | child   |
======================
| :person3 | :child6 |
| :person2 | :child5 |
| :person2 | :child4 |
| :person1 | :child3 |
| :person1 | :child2 |
| :person1 | :child1 |
----------------------

按照您的问题遍历结果将产生您当前获得的输出类型。我们想要做的是实际得到如下结果:

$ arq --data data.n3 --query query.sparql
----------------------------------------
| person   | child                     |
========================================
| :person3 | :child6                   |
| :person2 | :child4, :child5          |
| :person1 | :child1, :child2, :child3 |
----------------------------------------

这正是group_by 让我们做的事情。像这样的查询:

prefix : <http://example.org/people/>

select ?person (group_concat(?child;separator=' and ') as ?children) where { 
  ?person :hasChild ?child .
}
group by ?person

产生(注意结果中的变量是?children,而不是?child,因为我们使用group_concat(...) as ?children创建了新变量?children):

$ arq --data data.n3 --query query.sparql
---------------------------------------------------------------------------------------------------------------------------
| person   | children                                                                                                     |
===========================================================================================================================
| :person3 | "http://example.org/people/child6"                                                                           |
| :person1 | "http://example.org/people/child3 and http://example.org/people/child2 and http://example.org/people/child1" |
| :person2 | "http://example.org/people/child5 and http://example.org/people/child4"                                      |
---------------------------------------------------------------------------------------------------------------------------

如果您使用这样的查询并遍历结果并按原样打印它们,您将获得所需的输出。如果您确实想从人和孩子身上剥离领先的http://example.org/people/,您将需要更多的字符串处理。例如,使用STRAFTER 删除http://example.org/people/ 前缀,您可以使用如下查询:

prefix : <http://example.org/people/>

select
 (strafter(str(?personX),"http://example.org/people/") as ?person)
 (group_concat(strafter(str(?child),"http://example.org/people/");separator=' and ') as ?children)
where { 
  ?personX :hasChild ?child .
}
group by ?personX

得到如下结果:

$ arq --data data.n3 --query query.sparql
----------------------------------------------
| person    | children                       |
==============================================
| "person3" | "child6"                       |
| "person2" | "child5 and child4"            |
| "person1" | "child3 and child2 and child1" |
----------------------------------------------

当您进行打印时,它会为您提供类似的结果

person3 has children child6
person2 has children child5 and child4
person1 has children child3 and child2 and child1

【讨论】:

  • 非常棒。非常感谢你。你知道 Prolog 中是否有 group_concat 和 strafter 的等价物吗?
  • @user2598997 在纯 Prolog 中?或者您是否使用 Prolog 引擎来查询 RDF(例如,就像 Allego 提供的一样)?
  • 你说得对,我在 allegrograph 中使用 prolog: (select(?person ?child)(q ?person !ns:childOf ?child ))
  • @user2598997 嗯,除了很久以前的一些非常基本的用法,我对 AllegroGraph 不太熟悉。我怀疑它 支持 SPARQL 查询,但你也可以在它的 Prolog 中做一些事情。不过,这确实是一个不同的问题,我建议您将其作为一个单独的问题发布,以便获得更多可见性。如果您想将其表述为“我可以在 AllegroGraph 序言中执行 this 吗?”,您可以参考它。
  • 是的,它确实支持 SPARQL 查询,我现在将使用它。再次感谢您。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-07-19
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2017-09-02
  • 1970-01-01
相关资源
最近更新 更多