Я использую онтологию SUMO, которую хочу запросить с помощью SPARQL. Типичная запись в SUMO, например, для города, выглядит так:
<owl:Thing rdf:ID="MadridSpain">
<rdfs:isDefinedBy rdf:resource="http://www.ontologyportal.org/SUMO.owl"/>
<rdf:type rdf:resource="#City"/>
<owl:comment xml:lang="en">The City of Madrid in Spain.</owl:comment>
<geographicSubregion rdf:resource="#Spain" />
<externalImage rdf:datatype="xsd:anyURI">[...]</externalImage>
<rdfs:label xml:lang="en">madrid spain</rdfs:label>
</owl:Thing>
Если я хочу получить все города из онтологии, я использую этот пример запроса (который отлично работает):
String prefix = "PREFIX rdfs: <http://www.w3.org/2000/01/rdf-schema#> "
+ "PREFIX rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#>";
String rdq = prefix + "SELECT ?N ?O WHERE {?N rdf:type <http://www.ontologyportal.or/SUMO.owl#City>}";
Моя проблема начинается, когда я хочу отфильтровать результаты. Предположим, мне нужны только все города, которые являются географическим субрегионом Испании. Сначала я попытался решить эту проблему, проанализировав все результаты в Java и Jena, на что уходит огромное количество времени (5-10 секунд на каждый результат, всего ~ 10000 результатов).
Query myQuery = QueryFactory.create(rdq);
QueryExecution qexec = QueryExecutionFactory.create(myQuery, owlModel);
try {
ResultSet results = qexec.execSelect();
for (; results.hasNext();) {
QuerySolution sol = results.nextSolution();
Resource res = sol.getResource("N");
StmtIterator it = res.listProperties();
while(it.hasNext()){
Statement state = it.next();
//Doing some filtering
System.out.println("predicate: " + state.getPredicate().toString());
System.out.println("subject: " + state.getSubject().toString());
System.out.println("object: " + state.getObject().toString());
}
}
}catch (Exception e) {
e.printStackTrace();
System.err.println("Query Error " + e.getMessage());
}
Конечно, это не очень эффективно, и должен существовать более простой способ с использованием правильного запроса. Но на данный момент я застрял в определении такого запроса. Я пробовал следующие, но ни один из них не работает.
SELECT ?N ?O WHERE { ?N rdf:type <http://www.ontologyportal.org/SUMO.owl#City> .
{ SELECT ?N WHERE { (rdf:type ?b rdf:statement) .
(rdf:Predicate ?b <http://www.ontologyportal.org/SUMO.owl#geographicSubregion>) .
(rdf:Object ?b <http://www.ontologyportal.org/SUMO.owl#Spain>) } } }
SELECT ?N ?O WHERE { (rdf:statement ?b) .
(rdf:Predicate ?b <http://www.ontologyportal.org/SUMO.owl#geographicSubregion>) .
(rdf:Object ?b <http://www.ontologyportal.org/SUMO.owl#Spain>) . }";
Есть ли у кого-нибудь идея, как создать запрос, который получает все города в стране?