Итерация и инструкция FLWOR (XQuery)
Язык XQuery определяет синтаксис итераций инструкции FLWOR. Слово FLWOR — это сокращение от слов for
, let
, where
, order by
и return
. В SQL Server 2005 предложение let
не поддерживается.
Инструкция FLWOR состоит из следующих частей.
- Одно или несколько предложений FOR, которые привязывают одну или несколько переменных-итераторов к входным последовательностям.
Входные последовательности также могут быть выражениями XQuery (например выражениями XPath). Они являются либо последовательностями узлов, либо последовательностями атомарных значений. Последовательности атомарных значений могут быть получены с помощью литералов или функций-конструкторов. В SQL Server 2005 нельзя использовать построенные XML-узлы в качестве входных последовательностей. - Переменная-итератор. Дополнительно для этой переменной с помощью ключевого слова
as
можно указать тип. - Необязательное предложение
where
. Это предложение применяется в качестве предиката фильтра при итерации. - Необязательное предложение
order by
. - Выражение
return
. Выражение в предложенииreturn
конструирует результат, возвращаемый инструкцией FLWOR.
Например следующий запрос выполняет итерацию элементов <Step
> первого производства и возвращает строковое значение узлов <Step
>:
declare @x xml
set @x='<ManuInstructions ProductModelID="1" ProductModelName="SomeBike" >
<Location LocationID="L1" >
<Step>Manu step 1 at Loc 1</Step>
<Step>Manu step 2 at Loc 1</Step>
<Step>Manu step 3 at Loc 1</Step>
</Location>
<Location LocationID="L2" >
<Step>Manu step 1 at Loc 2</Step>
<Step>Manu step 2 at Loc 2</Step>
<Step>Manu step 3 at Loc 2</Step>
</Location>
</ManuInstructions>'
SELECT @x.query('
for $step in /ManuInstructions/Location[1]/Step
return string($step)
')
Результат:
Manu step 1 at Loc 1 Manu step 2 at Loc 1 Manu step 3 at Loc 1
Следующий запрос похож на предыдущий, за тем исключением, что он применяется к типизированному XML-столбцу Instructions в таблице ProductModel. Запрос перебирает все производственные этапы, то есть элементы <step
>, выполняемые в первом цехе для заданного изделия.
SELECT Instructions.query('
declare namespace AWMI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
for $Step in //AWMI:root/AWMI:Location[1]/AWMI:step
return
string($Step)
') as Result
FROM Production.ProductModel
where ProductModelID=7
В приведенном запросе обратите внимание на следующие моменты:
- переменная
$Step
является переменной-итератором; - выражение пути
//AWMI:root/AWMI:Location[1]/AWMI:step
формирует входную последовательность, то есть последовательность из дочерних элементов <step
> первого узлового элемента <Location
>; - необязательное предложение-предикат
where
не указано; - выражение
return
возвращает строковое значение элемента <step
>.
Функция string (XQuery) применяется для получения строкового значения узла <step
>.
Частичный результат:
Insert aluminum sheet MS-2341 into the T-85A framing tool.
Attach Trim Jig TJ-26 to the upper and lower right corners of
the aluminum sheet. ....
Ниже приведены примеры других допустимых входных последовательностей:
declare @x xml
set @x=''
SELECT @x.query('
for $a in (1, 2, 3)
return $a')
-- result = 1 2 3
declare @x xml
set @x=''
SELECT @x.query('
for $a in
for $b in (1, 2, 3)
return $b
return $a')
-- result = 1 2 3
declare @x xml
set @x='<ROOT><a>111</a></ROOT>'
SELECT @x.query('
for $a in (xs:string( "test"), xs:double( "12" ), data(/ROOT/a ))
return $a')
-- result test 12 111
В SQL Server 2005 гетерогенные последовательности не допускаются. В частности, недопустимы последовательности, содержащие и атомарные значения, и узлы.
Итерация часто применяется совместно с синтаксисом построения XML в трансформирующих XML-форматах, как показано в следующем запросе.
В образце базы данных AdventureWorks производственные инструкции хранятся в столбце Instructions таблицы Production.ProductModel в следующем формате:
<Location LocationID="10" LaborHours="1.2"
SetupHours=".2" MachineHours=".1">
<step>describes 1st manu step</step>
<step>describes 2nd manu step</step>
...
</Location>
...
Следующий запрос создает новый XML-документ, содержащий элементы <Location
> с атрибутами цеха в качестве дочерних элементов:
<Location>
<LocationID>10</LocationID>
<LaborHours>1.2</LaborHours>
<SetupHours>.2</SteupHours>
<MachineHours>.1</MachineHours>
</Location>
...
Запрос:
SELECT Instructions.query('
declare namespace AWMI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
for $WC in /AWMI:root/AWMI:Location
return
<Location>
<LocationID> { data($WC/@LocationID) } </LocationID>
<LaborHours> { data($WC/@LaborHours) } </LaborHours>
<SetupHours> { data($WC/@SetupHours) } </SetupHours>
<MachineHours> { data($WC/@MachineHours) } </MachineHours>
</Location>
') as Result
FROM Production.ProductModel
where ProductModelID=7
В приведенном запросе обратите внимание на следующие моменты:
- инструкция FLWOR получает для указанного изделия последовательность элементов <
Location
>; - функция data (XQuery) применяется для извлечения значения каждого из атрибутов, чтобы добавить его в результирующий XML-документ в виде текстовых узлов, а не атрибутов;
- выражение в предложении RETURN конструирует требуемый XML-документ.
Частичный результат:
<Location>
<LocationID>10</LocationID>
<LaborHours>2.5</LaborHours>
<SetupHours>0.5</SetupHours>
<MachineHours>3</MachineHours>
</Location>
<Location>
...
<Location>
...
Применение предложения where
Предложение where
может использоваться для фильтрации результатов итерации. Это продемонстрировано ниже на примере образца базы данных AdventureWorks.
При изготовлении велосипеда производственный процесс проходит через цепочку цехов. Для каждого цеха определена последовательность производственных операций. Следующий запрос извлекает только те цеха, которые занимаются изготовлением модели велосипеда и включают менее трех производственных операций, то есть содержат менее трех элементов <step
>.
SELECT Instructions.query('
declare namespace AWMI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
for $WC in /AWMI:root/AWMI:Location
where count($WC/AWMI:step) < 3
return
<Location >
{ $WC/@LocationID }
</Location>
') as Result
FROM Production.ProductModel
where ProductModelID=7
При рассмотрении предыдущего запроса необходимо отметить следующее:
- в ключевом слове
where
используется функция count(), подсчитывающая число дочерних элементов <step
> для каждого цеха; - выражение
return
конструирует XML-документ на основании результатов итерации.
Результат:
<Location LocationID="30"/>
Результат выражения в предложении where
преобразуется в логическое значение по описанным ниже правилам (в указанном порядке). Это те же правила, которые применяются к предикатам в выражениях пути, за исключением того, что в данном случае целочисленные значения недопустимы.
- Если выражение
where
возвращает пустую последовательность, действительное логическое значение равно False. - Если выражение
where
возвращает одно значение простого логического типа, результатом является это действительное логическое значение. - Если выражение
where
возвращает последовательность, в которой содержится хотя бы один узел, действительное логическое значение равно True. - В остальных случаях выдается статическая ошибка.
Множественная привязка переменных в инструкции FLWOR
В одном выражении инструкции FLWOR можно привязывать к входным последовательностям несколько переменных. В следующем примере выполняется запрос к нетипизированной переменной типа xml. Выражение FLOWR возвращает первый дочерний элемент <Step
> в каждом элементе <Location
>.
declare @x xml
set @x='<ManuInstructions ProductModelID="1" ProductModelName="SomeBike" >
<Location LocationID="L1" >
<Step>Manu step 1 at Loc 1</Step>
<Step>Manu step 2 at Loc 1</Step>
<Step>Manu step 3 at Loc 1</Step>
</Location>
<Location LocationID="L2" >
<Step>Manu step 1 at Loc 2</Step>
<Step>Manu step 2 at Loc 2</Step>
<Step>Manu step 3 at Loc 2</Step>
</Location>
</ManuInstructions>'
SELECT @x.query('
for $Loc in /ManuInstructions/Location,
$FirstStep in $Loc/Step[1]
return
string($FirstStep)
')
В приведенном запросе обратите внимание на следующие моменты:
- выражение
for
определяет переменные$Loc
и $FirstStep
; - выражения
/ManuInstructions/Location
и$FirstStep in $Loc/Step[1]
соотносятся так, что значения$FirstStep
зависят от значений$Loc
; - выражение, связанное с переменной
$Loc
, формирует последовательность из элементов <Location
>. Для каждого элемента <Location
> переменная$FirstStep
формирует последовательность из одного элемента <Step
>, одноэлементное множество; - переменная
$Loc
указана в выражении, связанном с переменной$FirstStep
.
Результат:
Manu step 1 at Loc 1
Manu step 1 at Loc 2
Следующий запрос аналогичен предыдущему, за исключением того, что он указан для столбца Instructions, типизированного столбца xml в таблице ProductModel. Для формирования необходимого XML-документа применяется конструкция XML (XQuery).
SELECT Instructions.query('
declare default namespace ="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
for $WC in /root/Location,
$S in $WC/step
return
<Step LocationID= "{$WC/@LocationID }" >
{ $S/node() }
</Step>
') as Result
FROM Production.ProductModel
WHERE ProductModelID=7
Для предыдущего запроса необходимо отметить следующее:
- Предложение
for
определяет две переменные:$WC
и$S
. выражение, связанное с$WC
, формирует последовательность цехов, участвующих в производстве велосипедов. Выражение пути, присвоенное переменной$S
, формирует последовательность технологических операций для каждого цеха, содержащегося в переменной$WC
; - инструкция return конструирует XML-документ, содержащий элемент <
Step
>, который соответствует технологической операции и имеет атрибут LocationID; - объявление declare default element namespace в прологе XQuery предназначено для того, чтобы все объявления пространства имен в результирующем XML-документе находились рядом с элементом верхнего уровня. Это повышает удобочитаемость результата. Дополнительные сведения о пространствах имен по умолчанию см. в разделе Поддержка пространств имен в XQuery.
Частичный результат:
<Step xmlns=
"https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions"
LocationID="10">
Insert <material>aluminum sheet MS-2341</material> into the <tool>T-
85A framing tool</tool>.
</Step>
...
<Step xmlns=
"https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions"
LocationID="20">
Assemble all frame components following blueprint
<blueprint>1299</blueprint>.
</Step>
...
Применение предложения order by
В языке XQuery сортировка в выражении FLWOR выполняется с помощью предложения order by
. Сортируемые выражения, передаваемые предложению order by
, должны возвращать значения, типы которых допустимы для оператора gt. Каждое сортируемое выражение должно возвращать последовательность, состоящую из одного элемента. По умолчанию сортировка выполняется в порядке возрастания. Для каждого из сортируемых выражений можно указать сортировку по возрастанию или по убыванию.
Примечание. |
---|
При сортировке сравнение символьных значений, выполняемых реализацией языка XQuery в SQL Server, всегда выполняется с использованием параметров сортировки двоичных элементов кода Юникода. |
Следующий запрос получает все телефонные номера указанного заказчика из столбца AdditionalContactInfo. Результат сортируется по номеру телефона.
SELECT AdditionalContactInfo.query('
declare namespace act="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ContactTypes";
declare namespace aci="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ContactInfo";
for $a in /aci:AdditionalContactInfo//act:telephoneNumber
order by $a/act:number[1] descending
return $a
') As Result
FROM Person.Contact
WHERE ContactID=3
Обратите внимание, что процесс атомизации (XQuery) получает атомарное значение элемента <number
>, прежде чем передать его предложению order by
. Выражение может содержать функцию data(), но ее использование необязательно.
order by data($a/act:number[1]) descending
Результат:
<act:telephoneNumber xmlns:act="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ContactTypes">
<act:number>333-333-3334</act:number>
</act:telephoneNumber>
<act:telephoneNumber xmlns:act="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ContactTypes">
<act:number>333-333-3333</act:number>
</act:telephoneNumber>
Вместо объявления пространства имен в прологе запроса можно объявить его с помощью предложения WITH XMLNAMESPACES.
WITH XMLNAMESPACES (
'https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ContactTypes' AS act,
'https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ContactInfo' AS aci)
SELECT AdditionalContactInfo.query('
for $a in /aci:AdditionalContactInfo//act:telephoneNumber
order by $a/act:number[1] descending
return $a
') As Result
FROM Person.Contact
WHERE ContactID=3
Сортировка может также выполняться по значению атрибута. Например: следующий запрос извлекает вновь созданные элементы <Location
>, имеющие атрибуты LocationID и LaborHours, сортируя результат по атрибуту LaborHours в убывающем порядке. В результате цеха, имеющие максимальное число рабочих часов, будут возвращены первыми.
SELECT Instructions.query('
declare namespace AWMI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
for $WC in /AWMI:root/AWMI:Location
order by $WC/@LaborHours descending
return
<Location>
{ $WC/@LocationID }
{ $WC/@LaborHours }
</Location>
') as Result
FROM Production.ProductModel
WHERE ProductModelID=7
Результат:
<Location LocationID="60" LaborHours="4"/>
<Location LocationID="50" LaborHours="3"/>
<Location LocationID="10" LaborHours="2.5"/>
<Location LocationID="20" LaborHours="1.75"/>
<Location LocationID="30" LaborHours="1"/>
<Location LocationID="45" LaborHours=".5"/>
В следующем запросе результат сортируется по имени элемента. Запрос извлекает характеристики указанного продукта из каталога продукции. Характеристики являются дочерними элементами элемента <Specifications
>.
SELECT CatalogDescription.query('
declare namespace
pd="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
for $a in /pd:ProductDescription/pd:Specifications/*
order by local-name($a)
return $a
') as Result
FROM Production.ProductModel
where ProductModelID=19
В приведенном запросе обратите внимание на следующие моменты:
- выражение
/p1:ProductDescription/p1:Specifications/*
возвращает дочерние элементы <Specifications
>; - выражение
order by (local-name($a))
сортирует последовательность по локальной части имени элемента.
Результат:
<Color>Available in most colors</Color>
<Material>Almuminum Alloy</Material>
<ProductLine>Mountain bike</ProductLine>
<RiderExperience>Advanced to Professional riders</RiderExperience>
<Style>Unisex</Style>
Узлы, в которых сортируемые выражения возвращают пустую последовательность, будут помещены в начало, как показано в следующем примере:
declare @x xml
set @x='<root>
<Person Name="A" />
<Person />
<Person Name="B" />
</root>
'
select @x.query('
for $person in //Person
order by $person/@Name
return $person
')
Результат:
<Person />
<Person Name="A" />
<Person Name="B" />
Следующий пример показывает, каким образом можно указать несколько критериев сортировки. Запрос в этом примере сортирует элементы <Employee
> сначала по значению атрибута Title, а затем по значению атрибута Administrator.
declare @x xml
set @x='<root>
<Employee ID="10" Title="Teacher" Gender="M" />
<Employee ID="15" Title="Teacher" Gender="F" />
<Employee ID="5" Title="Teacher" Gender="M" />
<Employee ID="11" Title="Teacher" Gender="F" />
<Employee ID="8" Title="Administrator" Gender="M" />
<Employee ID="4" Title="Administrator" Gender="F" />
<Employee ID="3" Title="Teacher" Gender="F" />
<Employee ID="125" Title="Administrator" Gender="F" /></root>'
SELECT @x.query('for $e in /root/Employee
order by $e/@Title ascending, $e/@Gender descending
return
$e
')
Результат:
<Employee ID="8" Title="Administrator" Gender="M" />
<Employee ID="4" Title="Administrator" Gender="F" />
<Employee ID="125" Title="Administrator" Gender="F" />
<Employee ID="10" Title="Teacher" Gender="M" />
<Employee ID="5" Title="Teacher" Gender="M" />
<Employee ID="11" Title="Teacher" Gender="F" />
<Employee ID="15" Title="Teacher" Gender="F" />
<Employee ID="3" Title="Teacher" Gender="F" />
Ограничения реализации
Существуют следующие ограничения:
- в выражении FLWOR не поддерживается предложение
let
; - Сортируемые выражения должны быть однородно типизированы. Эта проверка выполняется статически;
- нельзя управлять сортировкой пустых последовательностей;
- ключевые слова empty least, empty greatest и collation в выражении
order by
не поддерживаются.