=> Главная База Знаний Xslt Работа с переменными


Работа с переменными

Работа с переменными

Давайте рассмотрим примеры применения переменных. В следующем примере (листинг 9.1) я присваиваю переменной

copyright
сообщение об авторских правах и затем с ее помощью добавляю атрибут
copyright
во все элементы
planets.xml
.

Листинг 9.1. Применение переменной

<?xml version="1.0"?>

<xsl:stylesheet version="1.1"

 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

 <xsl:output method="xml"/>

 <xsl:variable name="copyright" select="'(c)2002 Starpowder Inc.'"/>

 <xsl:template match="*">

  <xsl:copy>

   <xsl:attribute name="copyright">

    <xsl:value-of select="$copyright"/>

   </xsl:attribute>

   <xsl:apply-templates/>

  </xsl:copy>

 </xsl:template>

</xsl:stylesheet>

Вот результирующий документ, дополненный атрибутами copyright

<?xml version="1.0" encoding="utf-8"?>

<PLANETS copyright="(c)2002 Starpowder Inc.">

 <PLANET copyright="(c)2002 Starpowder Inc.">

  <NAME copyright="(c)2002 Starpowder Inc.">Mercury</NAME>

  <MASS copyright="(c)2002 Starpowder Inc.">.0553</MASS>

  <DAY copyright="(с)2002 Starpowder Inc.">58.65</DAY>

  <RADIUS copyright="(c)2002 Starpowder Inc.">1516</RADIUS>

  <DENSITY copyright="(c)2002 Starpowder Inc.">.983</DENSITY>

  <DISTANCE copyright="(с)2002 Starpowder Inc.">43.4</DISTANCE>

 </PLANET>

 <PLANET copyright="(c)2002 Starpowder Inc.">

  <NAME copyright="(c)2002 Starpowder Inc.">Venus</NAME>

  <MASS copyright="(c)2002 Starpowder Inc.">.815</MASS>

  <DAY copyright="(с)2002 Starpowder Inc.">116.75</DAY>

  <RADIUS copyright="(c)2002 Starpowder Inc.">3716</RADIUS>

  <DENSITY copyright="(c)2002 Starpowder Inc.">.943</DENSITY>

  <DISTANCE copyright="(c)2002 Starpowder Inc.">66.8</DISTANCE>

 </PLANET>

 .

 .

 .

Переменные зачастую удобны для хранения значении, зависимых от контекста, и мы сейчас рассмотрим еще один пример, о котором я упоминал в начале главы. В этом случае я преобразую

planets.xml
в новый документ, в котором для каждой планеты будет один элемент. Каждый из этих новых элементов будет содержать два элемента
<SIBLINGPLANET>
, содержащих планеты-братья текущей планеты — например, братьями Земли будут Венера и Меркурий:

<?xml version="1.0" encoding="utf-8"?>

<Mercury>

 <SIBLINGPLANET>

  Venus

 </SIBLINGPLANET>

 <SIBLINGPLANET>

  Earth

 </SIBLINGPLANET>

</Mercury>

<Venus>

 <SIBLINGPLANET>

  Mercury

 </SIBLINGPLANET>

 <SIBLINGPLANET>

  Earth

 </SIBLINGPLANET>

</Venus>

<Earth>

 <SIBLINGPLANET>

  Mercury

 </SIBLINGPLANET>

 <SIBLINGPLANET>

  Venus

 </SIBLINGPLANET>

</Earth>

Для примера я поочередно выбираю каждый элемент

<PLANET>
и прохожу в цикле
<xsl:for-each>
по всем планетам, создавая элементы
<SIBLINGPLANET>
для всех планет, не являющихся контекстным узлом. Однако откуда мне известно внутри элемента
<xsl:for-each>
, какая из планет является контекстным узлом, выбранным шаблоном? Внутри элемента
<xsl:for-each>
«.» ссылается на текущий узел, с которым работает
<xsl:for-each>
, но не на контекстный узел шаблона. Проблему можно решить, если сохранить контекстный узел в переменной, которую я назвал
contextnode
:

<?xml version="1.0"?>

<xsl:stylesheet version="1.1"

 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

 <xsl:output method="xml"/>

 <xsl:template match="PLANETS">

  <xsl:for-each select="PLANET">

   <xsl:element name="{NAME}">

    <xsl:variable name="contextnode" select="."/>

    .

    .

    .

Теперь для проверки в цикле

<xsl:for-each>
того, что текущий элемент не является контекстным узлом, я могу обратиться к контекстному узлу шаблона как
$contextnode
(листинг 9.2).

Листинг 9.2. Хранение в переменной информации, зависимой от контекста

<?xml version="1.0"?>

<xsl:stylesheet version="1.1"

xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

<xsl:output method="xml"/>

 <xsl:template match="PLANETS">

  <xsl:for-each select="PLANET">

   <xsl:element name="{NAME}">

    <xsl:variable name="contextnode" select="."/>

    <xsl:for-each select="//PLANET">

     <xsl:if test=". != $contextnode">

      <xsl:element name="SIBLINGPLANET">

       <xsl:value-of select="NAME"/>

      </xsl:element>

     </xsl:if>

    </xsl:for-each>

   </xsl:element>

  </xsl:for-each>

 </xsl:template>

</xsl:stylesheet>

Теперь наша проблема решена.

Если у элемента

<xsl:variable>
есть тело, он создает переменную, чье значение является фрагментом результирующего дерева. В следующем примере при помощи фрагмента результирующего дерева я задаю значение по умолчанию для атрибута
COLOR
(цвет), если значение для него уже не задано. Значение по умолчанию я устанавливаю в «
blue
» (голубой):

<xsl:variable name="COLOR">

 <xsl:choose>

  <xsl:when test="@COLOR">

   <xsl:value-of select="@COLOR"/>

  </xsl:when>

  <xsl:otherwise>blue</xsl:otherwise>

 </xsl:choose>

</xsl:variable>

Строковое значение фрагмента результирующего дерева (то есть либо значение атрибута

COLOR
, либо значение по умолчанию, «
blue
») присваивается переменной
COLOR
. Теперь в выражениях XPath можно обращаться к значению этой переменной,
$COLOR
, а не к значению атрибута (
@COLOR
, гарантированно получая при этом значение цвета, даже если у соответствующего элемента отсутствует атрибут
COLOR
.

Вот еще один пример фрагмента результирующего дерева. В этом случае я сохраняю элемент буквального результата в переменной

START_HTML
:

<?xml version="1.0"?>

<xsl:stylesheet version="1.1"

 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

 <xsl:output method="html"/>

 <xsl:variable name="START_HTML">

  <HEAD>

   <TITLE>

    My page

   </TITLE>

  </HEAD>

 </xsl:variable>

 .

 .

 .

Теперь я могу использовать этот элемент буквального результата где угодно:

<?xml version="1.0"?>

<xsl:stylesheet version="1.1"

 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

 <xsl:output method="html"/>

 <xsl:variable name="START_HTML">

  <HEAD>

   <TITLE>

    My page

   </TITLE>

  </HEAD>

 </xsl:variable>


 <xsl:template match="PLANETS">

  <HTML>

   <xsl:copy-of select="$START HTML"/>

   <BODY>

    <H1>Welcome to my page</H1>

   </BODY>

  </HTML>

 </xsl:template>

</xsl:stylesheet>

И вот результат:

<HTML>

 <HEAD>

  <TITLE>

   My page

  </TITLE>

 </HEAD>

 <BODY>

  <H1>Welcome to my page</H1>

 </BODY>

</HTML>

Однако поскольку теперь фрагменты результирующего дерева не допускаются в XSLT 1.1, этот пример работать не будет. Как же тогда сохранить весь элемент буквального результата одновременно с возможностью простого вызова? Вы можете создать именованный шаблон.