Элементы XQuery, сгруппированные по тегам

Имея этот XML-файл:

<?xml version="1.0" encoding="ISO-8859-1"?>
<productos>
<TITULO>DATOS DE LA TABLA PRODUCTOS</TITULO>
<producto>
    <cod_prod>1010</cod_prod>
    <denominacion>Placa Base MSI G41M-P26</denominacion>
    <precio>50</precio>
    <stock_actual>10</stock_actual>
    <stock_minimo>3</stock_minimo>
    <cod_zona>10</cod_zona>
  </producto>
  <producto>
    <cod_prod>1011</cod_prod>
    <denominacion>Micro Intel Core i5-2320</denominacion>
    <precio>120</precio>
    <stock_actual>3</stock_actual>
    <stock_minimo>5</stock_minimo>
    <cod_zona>10</cod_zona>
  </producto>
  <producto>
    <cod_prod>1012</cod_prod>
    <denominacion>Micro Intel Core i5 2500</denominacion>
    <precio>170</precio>
    <stock_actual>5</stock_actual>
    <stock_minimo>6</stock_minimo>
    <cod_zona>20</cod_zona>
  </producto>
  <producto>
    <cod_prod>1013</cod_prod>
    <denominacion>HD Seagate Barracuda 250GB SATA</denominacion>
    <precio>80</precio>
    <stock_actual>10</stock_actual>
    <stock_minimo>5</stock_minimo>
    <cod_zona>20</cod_zona>
  </producto>
  <producto>
    <cod_prod>1014</cod_prod>
    <denominacion>HD Caviar Blue 500GB SATA3</denominacion>
    <precio>150</precio>
    <stock_actual>5</stock_actual>
    <stock_minimo>6</stock_minimo>
    <cod_zona>30</cod_zona>
  </producto>
  <producto>
    <cod_prod>1015</cod_prod>
    <denominacion>Tarjeta gráfica Asus GeForce EN210 Silent 1GB</denominacion>
    <precio>40</precio>
    <stock_actual>10</stock_actual>
    <stock_minimo>5</stock_minimo>
    <cod_zona>30</cod_zona>
  </producto>
  <producto>
    <cod_prod>1016</cod_prod>
    <denominacion>Tarjeta gráfica Gigabyte GeForce 1GB</denominacion>
    <precio>50</precio>
    <stock_actual>5</stock_actual>
    <stock_minimo>6</stock_minimo>
    <cod_zona>40</cod_zona>
  </producto>
  <producto>
    <cod_prod>1017</cod_prod>
    <denominacion>Tarjeta gráfica Nvidia Express 1GB</denominacion>
    <precio>45</precio>
    <stock_actual>10</stock_actual>
    <stock_minimo>5</stock_minimo>
    <cod_zona>30</cod_zona>
  </producto>
  <producto>
    <cod_prod>1018</cod_prod>
    <denominacion>Micro Intel Dual Core G620</denominacion>
    <precio>60</precio>
    <stock_actual>15</stock_actual>
    <stock_minimo>5</stock_minimo>
    <cod_zona>40</cod_zona>
  </producto>
  <producto>
    <cod_prod>1019</cod_prod>
    <denominacion>Memoria DDR3 G.Skill 2GB</denominacion>
    <precio>10</precio>
    <stock_actual>5</stock_actual>
    <stock_minimo>3</stock_minimo>
    <cod_zona>10</cod_zona>
  </producto>
  <producto>
      <cod_prod>1020</cod_prod>
      <denominacion>Memoria DDR3 G.Skill 4GB</denominacion>
      <precio>30</precio>
      <stock_actual>30</stock_actual>
      <stock_minimo>10</stock_minimo>
      <cod_zona>10</cod_zona>
  </producto>
  <producto>
    <cod_prod>1021</cod_prod>
    <denominacion>Memoria DDR3 Kingston HyperX 4GB</denominacion>
    <precio>16</precio>
    <stock_actual>15</stock_actual>
    <stock_minimo>4</stock_minimo>
    <cod_zona>20</cod_zona>
  </producto>
  <producto>
  <cod_prod>1022</cod_prod>
    <denominacion>Placa Base ASRock G41M-S3 </denominacion>
    <precio>52</precio>
    <stock_actual>2</stock_actual>
    <stock_minimo>2</stock_minimo>
    <cod_zona>30</cod_zona>
  </producto>
</productos>

Я хотел бы поместить "productos", чей "denominacion" является одинаковым внутри тегов. Например:

<placas> Placa Base MSI G41M-P26 Placa Base ASRock G41M-S3 </placas>
<micros> Micro Intel Core i5-2320 Micro Intel Core i5 2500 Micro Intel Dual Core G620 </micros>

За исключением тех, которые начинаются с "HD" и "Tarjeta" (которые должны быть внутри тега: "otros"). Вот пример того, что я сделал:

for $productos in doc("productos.xml")/productos/producto
let $agrupacion := $productos/substring-before(data(denominacion),' ')
group by $agrupacion
order by $agrupacion
return if (contains($agrupacion,'Placa'))
then <placa>{data($productos/denominacion)}</placa>
else if (contains($agrupacion,'Micro'))
then <micro>{data($productos/denominacion)}</micro>
else if (contains($agrupacion,'Memoria'))
then <memoria>{data($productos/denominacion)}</memoria>
else <otros>{data($productos/denominacion)}</otros>

Заранее спасибо.

1 ответ

Решение

Далее используется простая функция для сопоставления деноминации с именем строки / элемента, а затем выполняется группировка:

declare function local:map ($denominacion as xs:string) as xs:string {
  if (contains($denominacion, 'Placa'))
  then 'placa'
  else if (contains($denominacion, 'Memoria'))
  then 'memoria'
  else if (contains($denominacion, 'Micro'))
  then "micro"
  else 'otros'
 };

for $producto in /productos/producto
group by $denominacion := local:map($producto/denominacion)
order by $denominacion
return element {$denominacion} {$producto/denominacion/data()}

таким образом я получаю

<memoria>Memoria DDR3 G.Skill 2GB Memoria DDR3 G.Skill 4GB Memoria DDR3 Kingston HyperX 4GB</memoria>
<micro>Micro Intel Core i5-2320 Micro Intel Core i5 2500 Micro Intel Dual Core G620</micro>
<otros>HD Seagate Barracuda 250GB SATA HD Caviar Blue 500GB SATA3 Tarjeta gráfica Asus GeForce EN210 Silent 1GB Tarjeta gráfica Gigabyte GeForce 1GB Tarjeta gráfica Nvidia Express 1GB</otros>
<placa>Placa Base MSI G41M-P26 Placa Base ASRock G41M-S3 </placa>
Другие вопросы по тегам