запустить xpath из xquery для базы данных? - PullRequest
0 голосов
/ 01 января 2019

Как запустить запрос xpath из сценария xquery с basex GUI?

успешный xpath запрос базы данных: xpath

не удалось xquery попытка: xquery

Простейший xquery код для этой базы данных:

thufir@dur:~/basex$ 
thufir@dur:~/basex$ cat db_list_items.xq 

let $db := db:open("list")

return root()/descendant::li/a/text()

thufir@dur:~/basex$ 

Не совсем уверенкак вернуть результаты выше.

журнал ошибок из basex GUI:

Error:
Stopped at /home/thufir/basex/db_list_items.xq, 4/12:
[XPDY0002] root(): no context value bound.
Compiling:
- pre-evaluate db:open(database[,path]) to document-node(): db:open("list") -> db:open-pre("list", 0)
- inline $db_0
- simplify gflwor
Optimized Query:
root()/descendant::li/a/text()
Query:
let $db := db:open("list") return root()/descendant::li/a/text()
Query plan:
<QueryPlan compiled="true" updating="false">
  <CachedPath type="text()*">
    <FnRoot name="root([node])" type="node()?"/>
    <IterStep axis="descendant" test="li" type="element()*"/>
    <IterStep axis="child" test="a" type="element()*"/>
    <IterStep axis="child" test="text()" type="text()*"/>
  </CachedPath>
</QueryPlan>

ищет простой запрос к этой базе данных, используя basex.

1 Ответ

0 голосов
/ 15 февраля 2019

Точно так же, как навигация по файловой системе (или любой другой древовидной структуре), нам нужно знать наше место в этой структуре, чтобы иметь возможность использовать выражения пути:

/home/logname/documents/work #> cd todo
/home/logname/documents/work/todo #>

Как выполнялась командаЛинейный интерпретатор знает, где найти каталог todo ?Он знал это, потому что по соглашению предполагал, что контекстом для команды cd будет текущий каталог , в данном случае /home/logname/documents/work.

Когда вы загружаетедокумент / база данных в BaseX, он делает то же самое для выражений XPath и XQuerys, если контекст понятен.Если, например, вы поместите одну точку . в редакторе запросов, а затем выполните этот запрос, он вернет вам весь документ, поскольку именно это означает .: текущий элемент контекста .Здесь BaseX знает этот контекст из соглашения, что он по умолчанию выполняет запросы к текущему загруженному документу, точно так же, как интерпретатор командной строки принимает текущий каталог в качестве текущего контекста.Можно сказать, что, загружая документ / базу данных, вы cd 'врезались в корень этого документа.Пока все хорошо ...

Как только вы используете XQuery, вы используете полный язык программирования, который позволяет делать гораздо больше, чем просто запрашивать отдельные документы.Вы можете запросить целую кучу документов в одном скрипте.

Взять этот (неполный) фрагмент кода:

let $db  := db:open("list")
let $db2 := db:open("list2")

Если вы теперь размещаете запросы так, как делали, куда они идут?Используют ли они $db в качестве контекста или $db2?

Что вам нужно сделать, это сообщить процессору об этом.Это можно сделать несколькими способами:

  1. в прологе вашего скрипта: declare context item := db:open("list"); (см. Также: Документация BaseX по этому и, что очень важно,, читайте о разнице между "статическим контекстом" и "динамическим контекстом" здесь ), чтобы узнать больше о контексте в целом.
  2. в самом выражении XPath:
let $db := db:open("list")
return $db/root()/descendant::li/a/text()

или, упрощенно:

let $db := db:open("list")
return $db//li/a/text()

или:

declare context item := db:open("list");

.//li/a/text()
...