XSLT — страница 36 из 124

Предикаты, третья часть образцов шага, содержат выражения XPath. Предикат можно заключить в операцию [] и проверить, верно ли заданное условие.

Например, можно проверить:

• значение атрибута в заданной строке;

• значение элемента;

• содержит ли элемент определенного ребенка, атрибут или другой элемент;

• позицию узла в дереве узлов.

Подробнее выражения XPath обсуждаются в главе 7, но здесь будет представлено введение в тему, так как эти выражения можно применять в предикатах образцов.

Выражения XPath более сложны, чем образцы выбора. Если при их создании у вас возникнут затруднения, вам может помочь удобная программа-пример ApplyXPath.java из пакета Xalan, при помощи которой можно применить к документу выражение XPath и посмотреть на результаты. Например, если применить выражение XPath "

PLANET/NAME
" к
planets.xml
, будут отображены значения всех элементов
, дочерних по отношению к элементам
(открывающий и закрывающий теги
добавляются программой ApplyXPath):

C:\>java ApplyXPath planets.xml PLANET/NAME

Mercury

Venus

Earth

Если предикат имеет числовое значение, последнее представляет условие позиции (position test). Например, NAME[1] выбирает первого ребенка

контекстного узла. Условия позиции W3C, а также условия позиции в Xalan, Oracle, XT, Saxon и MSXML3 (XML процессор от Microsoft, подразумевающий использование JavaScript, с которым вы встречались в главе 1 и еще встретитесь в главе 10, «Работа с API процессоров XSLT») основаны на 1, поэтому первый ребенок — это ребенок 1. Условия позиции в XML-документах, которые используют таблицы стилей XSL и загружаются в текущую версию Internet Explorer (версию 5.5, на смену которой приходит 6.0), основаны на 0 (и в предикатах можно использовать только очень сокращенную форму выражений XPath) — и, следовательно, так же считается в большей части документации по XSL на web-узле Microsoft. В другом случае значением предиката должна быть ложь или истина, что называется логическим условием (Boolean test). Например, предикат
[@UNITS="million miles"]
выбирает элементы, у которых имеются атрибуты
UNITS
со значением "
million miles
".

Предикаты являются полными выражениями XPath, хотя на предикаты, используемые в образцах, накладывается два ограничения:

• когда образец используется в атрибуте

match
, предикат не должен содержать никаких ссылок на переменные XSL (которые обсуждаются в главе 9). Это ограничение не применяется к предикатам, используемым в элементах
;

• образцы не могут использовать в предикатах функцию XPath

current
, возвращающую текущий узел. Ее применение ограничено, поэтому обработка зависит от реализации и не зависит от текущего состояния обработки.

В следующем примере образец выбирает элементы

с дочерними элементами
:

 .

 .

 .

Этот образец выбирает любой элемент с дочерним элементом

:

 .

 .

 .

Теперь я задал элементам

в
planets.xml
новый атрибут,
COLOR
, устанавливающий цвет планеты:

Mercury

.0553

58.65

1516

.983

43.4

Venus

.815

116.75

3716

.943

66.8

Earth

1

2107

1

128.4

Следующее выражение выбирает элементы

с атрибутом
COLOR
:

 .

 .

 .

А что, если нам требуется выбрать планеты, у которых атрибут

COLOR
имеет значение "
BLUE
" (голубой)? Это можно сделать при помощи операции =, как показано в листинге 4.5.

Листинг 4.5. Применение операции =

<"xml version="1.0"?>

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


  The  is blue.


Таблица стилей из листинга 4.5 отбирает планеты с голубым цветом и убирает остальные, выключая правило по умолчанию для текстовых узлов. Результат следующий:

 The Earth is blue.

Создание предикатов

Предикаты — настоящие выражения XPath, и XPath гораздо ближе к настоящему языку, чем образцы: к примеру, выражения XPath могут возвращать не только списки узлов, но также логические, строковые и числовые значения. Выражения XPath могут работать не только с текущим узлом или дочерними узлами: можно работать с родительскими узлами, узлами-предками и другими узлами.

Глава 7 полностью посвящена XPath, но имеет смысл предоставить введение в предмет здесь, при обсуждении образцов, потому что часть предиката образца обладает наибольшими возможностями. В предикатах могут быть все виды выражений; в следующем списке перечислен ряд возможных типов, которые будут изучены в следующих разделах:

• наборы узлов;

• логические выражения;

• числа;

• строки.

Предикаты: наборы узлов

Набор узлов (node set), как понятно из названия, представляет собой просто совокупность узлов (и может содержать только один узел). Выражение

child::PLANET
возвращает набор узлов, состоящий из всех элементов
. Выражение
child::PLANET/child::NAME
возвращает список узлов, состоящий из всех элементов
, дочерних по отношению к элементам
. Для выбора узла или узлов из набора узлов воспользуйтесь следующими функциями для работы с наборами узлов в предикатах:

• 

last()
. Возвращает количество узлов в наборе узлов;

• 

position()
. Возвращает позицию контекстного узла в контекстном наборе узлов (начиная с 1);

• 

count(node-set)
. Возвращает количество узлов в наборе. Если опустить
node-set
, функция будет применена к контекстному узлу;

• 

id(string ID)
. Возвращает набор узлов, содержащий элемент с ID, удовлетворяющим переданной функции строке, или пустой набор узлов, если такой элемент отсутствует. Можно перечислить несколько идентификаторов, разделенных символами-разделителями, — тогда функция вернет набор узлов, состоящий из элементов с этими идентификаторами;

• 

local-name(node-set)
. Возвращает локальное имя первого узла в наборе узлов. Если опустить
node-set
, функция будет применена к контекстному узлу;

• 

namespace-uri(node-set)
. Возвращает URI пространства имен первого узла в наборе узлов. Если опустить
node-set
, функция будет применена к контекстному узлу;

• 

name(node-set)
. Возвращает полностью определенное имя первого узла в наборе узлов. Если опустить
node-set
, функция будет применена к контекстному узлу.

В листинге 4.6 я перенумеровал элементы в выходном документе при помощи функции

position()
.

Листинг 4.6. Применение функции position

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

</code></pre></p><p><pre><code>     The Planets</code></pre></p><p><pre><code>


Для вычисления логических результатов true/false можно применять следующие логические операции XPath:

• != означает «не равно»;

• < означает «меньше, чем» (в документах XML или XSL используйте <);

• <= означает «меньше или равно» (в документах XML или XSL используйте <=);

• = означает «равно» (программисты на С, С++, Java и JavaScript, обратите внимание: эта операция пишется как один знак =, а не два);

• > означает «больше, чем»;

• >= означает «больше или равно».

ИСПОЛЬЗОВАНИЕ СИМВОЛА <

Особенно обратите внимание на то, что непосредственно в документах XML или XSL нельзя использовать символ <, необходимо использовать ссылку на сущность <.

Для связи логических выражений логическими операциями And и Or используются ключевые слова and и or; слово not инвертирует логический смысл выражения — с истины на ложь или со лжи на истину.

В листинге 4.7 я определяю элемент

Земли и помещаю в таблицу строки "
Earth
", "
needs
", "
no
" и "
introduction
" вместо числовых данных Земли. Я определяю, которая из планет есть Земля, при помощи предиката
"[NAME='Earth']"
, проверяющего значение элемента
, которое, в свою очередь, представляет собой заключенный в элементе текст. Я также предоставил шаблон для других планет, удовлетворяющих предикату "
[NAME!='Earth']
''.

Листинг 4.7. Определение планеты Земля

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

</code></pre></p><p><pre><code>     The Planets Table</code></pre></p><p><pre><code>

     The Planets Table

NameMassRadiusDay


Earth

needs

no

introduction.





Вот результат:

</code></pre></p><p><pre><code>   The Planets Table</code></pre></p><p><pre><code>

<Н1>

   The Planets Table

   .

   .

   .

NameMassRadiusDay
Earthneedsnointroduction.

Результат можно увидеть на рис. 4.1.

Рис. 4.1. Применение предикатов XPath


В следующем примере используется логическая операция >. Это правило применяется ко всем элементам

после позиции 5:

Имеется также функция

true
, всегда возвращающая значение true, и функция
false
, всегда возвращающая значение false. Функция
not
инвертирует логический смысл выражения, как в следующем случае, где я выбираю все элементы
, кроме последнего:

Наконец, функция

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

Предикаты: числа

В XPath числа хранятся в формате числа с плавающей точкой двойной точности. (Технически все числа XPath хранятся в 64-разрядном формате IEEE числа с плавающей точкой двойной точности, floating-point double.) Все числа хранятся как числа с двойной точностью — даже целые числа, как 5 в рассматриваемом примере:

Над числами можно производить ряд операций:

• 

+
сложение;

• 

-
вычитание;

• 

*
умножение;

• 

div
деление (символ /, соответствующий делению в других языках, в XML, XSL и XPath уже занят);

• 

mod
возвращает значение деления по модулю двух чисел (остаток после деления первого числа на второе).

Например, элемент

вставит в выходной документ строку "
600
". В следующем примере выбираются все планеты, у которых отношение дня (измеренного в днях Земли) к массе (где масса Земли принята за 1) больше 100:

XPath также поддерживает следующие функции работы с числами:

• 

ceiling()
. Возвращает наименьшее целое, большее, чем переданное функции число;

• 

floor()
. Возвращает наибольшее целое, меньшее, чем переданное функции число;

• 

round()
. Округляет переданное число до ближайшего целого;

• 

sum()
. Возвращает сумму переданных функции чисел.

Например, среднюю массу планет в

planets.xml
можно найти так, как в листинге 4.8:

Листинг 4.8. Вычисление средней массы планет

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

    The average planetary mass is:

Строки

В XPath строки формируются из символов Unicode, как можно было предположить. Ряд функций специально предназначен для работы со строками:

• 

string(object object1)
. Преобразует объект в строку;

• 

starts-with(string string1, string string2)
. Возвращает истину, если первая строка начинается (starts with) со второй строки;

• 

contains(string string1, string string2)
. Возвращает истину, если первая строка содержит (contains) вторую строку;

• 

substring(string string1, number offset number length)
. Возвращает
length
символов из строки, начиная со смещения
offset
;

• 

substring-before(string string1, string string2)
. Возвращает часть строки
string1
до первого вхождения строки
string2
;

• 

substring-after(string string1, string string2)
. Возвращает часть строки
string1
после первого вхождения
string2
;

• 

string-length(string string1)
. Возвращает количество символов в строке
string1
;

• 

normalize-space(string string1)
. Возвращает строку
string1
после отбрасывания лидирующих и завершающих символов-разделителей и замены нескольких последовательных разделителей на один пробел;

• 

translate(string string1, string string2, string string3)
. Возвращает строку
string1
, в которой все вхождения символов в строке
string2
заменены на соответствующие символы в строке
string3
;

• 

concat(string string1, string string2, ...)
. Возвращает конкатенацию (объединение) всех строк.

Есть еще одна строковая функция, о которой вам следует знать, входящая не в XPath, а в XSLT:

• 

format-number(number number1, string string2, string string3)
. Возвращает строку, содержащую число
number1
в виде форматированной строки, используя
string2
в качестве форматирующей строки (форматирующие строки создаются так же, как для метода Java java.text.DecimalFormat) и
string3
как возможную строку локализации.

В листинге 4.9 я выбираю текстовые узлы, в которых текст начинается с 'Е', чтобы выбрать Earth (Земля), и добавляю текст '(the World)' (мир), получая 'Earth (the World)'. Для этого я применяю предикат "

text()[starts-with(., 'Е')]
".

Листинг 4.9. Применение функции starts-with

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

    .

    .

    .



(the World)


 .

 .

 .

А вот результат — заметьте, что заголовок для Земли стал "Earth (the World)":

</code></pre></p><p><pre><code>   The Planets Table</code></pre></p><p><pre><code>

   The Planets Table

NameMassRadiusDay
Earth (the World)1 (Earth = 1)2107 miles1 days

Этот документ показан на рис. 4.2.

Рис. 4.2. Применение текстовых предикатов 

Предикаты: фрагменты результирующего дерева

XSLT 1.0 добавляет к поддерживаемым XPath типам данных фрагменты результирующего дерева. Это фрагменты дерева, которые можно присваивать переменным XSLT, они не очень широко распространены. Практически все, что можно с ними сделать, — это вычислить их строковое значение. В рабочем проекте XSLT 1.1 их поддержка была удалена, поэтому, видимо, в XSLT 2.0 их уже не будет.

Сокращенный синтаксис предикатов

Выражения предикатов можно сокращать, опуская "

position()=
". Например,
[position()=3]
становится
[3]
,
[position()=last()]
становится
[last()]
и т.д. С использованием сокращенного синтаксиса применять выражения XPath в предикатах становится существенно проще. Вот ряд примеров:

• 

PLANET[2]
. Возвращает второго ребенка
контекстного узла;

• 

PLANET[last()]
. Возвращает последнего ребенка
контекстного узла;

• 

/PLANETS/PLANET[2]/NAME[1]
. Возвращает первый элемент
второго элемента
элемента
;

• 

PLANET[5][@UNITS="million miles"]
. Возвращает пятого ребенка
контекстного узла, только если у него имеется атрибут
UNITS
со значением "
million miles
". Это выражение можно также написать как
PLANET[@UNITS="million miles"][5]
.

На этом мы заканчиваем рассмотрение трех частей образцов шага: осей, условий узлов и предикатов. Это строительные блоки образцов выбора. Лучше всего изучить создание образцов на примере, и многие примеры мы вскоре рассмотрим. Сначала, однако, важно рассмотреть две небольшие темы. Как вы помните из формального определения образцов выбора, можно помимо образцов шага, создавать образцы, выбирающие элементы по идентификатору (ID) или ключу.

Выбор по ID