Поделиться через


Итерация и инструкция 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 преобразуется в логическое значение по описанным ниже правилам (в указанном порядке). Это те же правила, которые применяются к предикатам в выражениях пути, за исключением того, что в данном случае целочисленные значения недопустимы.

  1. Если выражение where возвращает пустую последовательность, действительное логическое значение равно False.
  2. Если выражение where возвращает одно значение простого логического типа, результатом является это действительное логическое значение.
  3. Если выражение where возвращает последовательность, в которой содержится хотя бы один узел, действительное логическое значение равно True.
  4. В остальных случаях выдается статическая ошибка.

Множественная привязка переменных в инструкции 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. Каждое сортируемое выражение должно возвращать последовательность, состоящую из одного элемента. По умолчанию сортировка выполняется в порядке возрастания. Для каждого из сортируемых выражений можно указать сортировку по возрастанию или по убыванию.

ms190945.note(ru-ru,SQL.90).gifПримечание.
При сортировке сравнение символьных значений, выполняемых реализацией языка 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 не поддерживаются.