Совместимы ли классы .NET 3.5 XPath и методы XSLT 2.0?
Вопрос
Я бы хотел использовать регулярные выражения при выборе элементов с помощью функции match.Я бы предпочел не использовать для этого внешнюю библиотеку (например, saxon).
Решение
В XSLT 2.0 есть некоторые вещи, которые не поддерживаются во встроенных библиотеках (в списке рассылки mono было обсуждение по этому поводу, но я больше не могу найти информацию).Но большинство людей никогда не сталкиваются с угловыми случаями, которые не поддерживаются.
Другой вариант - ознакомиться с открытым исходным кодом http://saxon.sourceforge.net/ который имеет отличную поддержку версии 2.0.
РЕДАКТИРОВАТЬ (AB):приведенный выше общепринятый ответ может ввести в заблуждение.Там вообще нет никакой поддержки, а есть никаких планов в этом направлении для любой из функций XPath 2.0 или XSLT 2.0 в .NET.
Другие советы
Я считаю, что ответ в этом обсуждении вводит в заблуждение.Я думаю, что .NET 3.5 не поддерживает большинство функций XSL / T 2.0 (если они вообще есть).
Пример:
Вызов функции 2.0 выдает следующее сообщение об ошибке в .NET 3.5:
'current-DateTime()' - это неизвестная функция XSLT.
Я думаю, что приведенный выше ответ неверен.Я не могу найти никаких доказательств того, что Microsoft поддерживает XSLT 2.0.XSLT != XPath.
Для дальнейшего использования, вот хорошая страница о расширении xpath / xquery в .net:
http://www.csharpfriends.com/Articles/getArticle.aspx?articleID=64
Я не верю, что это надолго, поэтому копирую это здесь:
XSLT - это язык преобразования XML.Это позволяет серверным системам преобразовывать исходное XML-дерево в более подходящую форму для клиентов.XSLT использует шаблоны узлов для сопоставления с шаблонами для выполнения своих преобразований.Хотя это делает сложные преобразования относительно простыми, есть некоторые ситуации, когда нам, возможно, придется использовать некоторые пользовательские классы.
Вот некоторые из ситуаций, в которых нам может потребоваться расширить XSLT:
1) Вызов пользовательской бизнес-логики
2) Выполнять различные действия в зависимости от разрешений
3) Выполните сложное форматирование дат, строк и т. Д
4) Или даже вызвать веб-сервис!!
Шаги по расширению XSLT
1) Создайте пользовательский объект для использования из XSLT (на C #)
CustomDate custDate = new CustomDate() ;
2) Предоставьте пользовательское объявление пространства имен для пользовательского класса в объявлении пространства имен XSLTs (в файле XSLT)
<xsl:transform
version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:myCustDate="urn:custDate">
3) Передайте экземпляр пользовательского объекта в XSLT с тем же пространством имен, что и на последнем шаге (в C #).
xslArgs.AddExtensionObject("urn:custDate", custDate) ;
4) Используйте объект из XSLT (в файле XSLT).
<xsl:value-of select="myCustDate:GetDateDiff(./joiningdate)"/>
Пример кода
Для нашего примера предположим, что у нас есть XSLT-лист, где нам нужно манипулировать датами.Нам нужно показать, сколько дней сотрудник проработал в компании.Поскольку XSLT не имеет собственных функций манипулирования датой, давайте используем объект расширения для нашей задачи.
using System ;
using System.IO ;
using System.Xml ;
using System.Xml.Xsl ;
using System.Xml.XPath ;
public class XsltExtension{
public static void Main(string[] args){
if (args.Length == 2){
Transform(args[0], args[1]) ;
}else{
PrintUsage() ;
}
}
public static void Transform(string sXmlPath, string sXslPath){
try{
//load the Xml doc
XPathDocument myXPathDoc = new XPathDocument(sXmlPath) ;
XslTransform myXslTrans = new XslTransform() ;
//load the Xsl
myXslTrans.Load(sXslPath) ;
XsltArgumentList xslArgs = new XsltArgumentList() ;
//create custom object
CustomDate custDate = new CustomDate() ;
//pass an instance of the custom object
xslArgs.AddExtensionObject("urn:custDate", custDate) ;
//create the output stream
XmlTextWriter myWriter = new XmlTextWriter("extendXSLT.html", null) ;
//pass the args,do the actual transform of Xml
myXslTrans.Transform(myXPathDoc,xslArgs, myWriter) ;
myWriter.Close() ;
}catch(Exception e){
Console.WriteLine("Exception: {0}", e.ToString());
}
}
public static void PrintUsage(){
Console.WriteLine("Usage: XsltExtension.exe <xml path> >xsl path<") ;
}
}
//our custom class
public class CustomDate{
//function that gets called from XSLT
public string GetDateDiff(string xslDate){
DateTime dtDOB = DateTime.Parse(xslDate) ;
DateTime dtNow = DateTime.Today ;
TimeSpan tsAge = dtNow.Subtract(dtDOB) ;
return tsAge.Days.ToString() ;
}
}
Скомпилируйте этот код и используйте предоставленные файлы members.xml и memberdisplay.xsl для запуска этого консольного приложения.Вы должны увидеть extendXSLT.html файл в той же папке.Откройте этот файл и обратите внимание, что наш класс CustomDate был вызван для вычисления количества дней, в течение которых сотрудник находился в компании.
Краткие сведения :
XSLT - мощный язык преобразования для XML, однако использование объектов расширения в .NET и C # должно гарантировать, что мы сможем легко выполнить то, что было бы невозможно или сложно с одним только XSLT.
Members.xml:
<root>
<member>
<name>Employee1</name>
<joiningdate>01/01/1970</joiningdate>
<role>CTO</role>
</member>
<member>
<name>Employee2</name>
<joiningdate>24/07/1978</joiningdate>
<role>Web Developer</role>
</member>
<member>
<name>Employee3</name>
<joiningdate>15/12/1980</joiningdate>
<role>Tester</role>
</member>
</root>
Memberdisplay.xsl:
<xsl:transform
version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:myCustDate="urn:custDate">
<xsl:output method="html" omit-xml-declaration="yes" />
<xsl:template match="/">
<html>
<head>
<style>
TABLE.tblMaster
{
border-style: solid;
border-width: 1px 1px 1px 1px;
border-style: solid;
border-color: #99CCCC;
padding: 4px 6px;
text-align: left;
font-family:Tahoma,Arial;
font-size:9pt;
}
TD.tdHeader
{
FONT-WEIGHT: bolder;
FONT-FAMILY: Arial;
BACKGROUND-COLOR: lightgrey;
TEXT-ALIGN: center
}
</style>
</head>
<body>
<table width="50%" class="tblMaster">
<tr >
<td class="tdHeader">Employee</td>
<td class="tdHeader">Join date</td>
<td class="tdHeader">Days in company</td>
<td class="tdHeader">Role</td>
</tr>
<xsl:for-each select="/root/member">
<tr >
<td> <xsl:value-of select="./name"/> </td>
<td> <xsl:value-of select="./joiningdate"/> </td>
<td> <xsl:value-of select="myCustDate:GetDateDiff(./joiningdate)"/> </td>
<td> <xsl:value-of select="./role"/> </td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:transform>
При обсуждении .СЕТЕВАЯ поддержка XSLT 2.0, XPath 2.0 и XQuery 1.0 важно проводить различие между самими языками и моделью данных (XDM).Платформа .NET 3.5 поддерживает модель данных, но не языки.Как мне недавно объяснил по электронной почте Павел Кадлучка из Microsoft:
Предложение "экземпляры модели данных XQuery 1.0 и XPath 2.0" может сбивать с толку, но я полагаю, что оно относится к данным W3C XQuery 1.0 и XPath 2.0 Спецификация модели (XDM) (http://www.w3.org/TR/xpath-datamodel) это гласит:
[Определение:Каждый экземпляр модели данных представляет собой последовательность.].
[Определение:Последовательность - это упорядоченная коллекция из нуля или более элементов.] Последовательность не может быть членом последовательности.Отдельный элемент, появляющийся сам по себе , моделируется как последовательность , содержащая один элемент.Последовательности определены в 2.5 последовательностях.
[Определение:Элемент - это либо узел , либо атомарное значение],
В случае XPath API - XPathNodeIterator - это последовательность, в то время как XPathItem (XPathNavigator) представляет элемент.
Да, 3.5 XPathNavigator поддерживает XSLT 2.0.
http://msdn.microsoft.com/en-us/library/system.xml.xpath.xpathnavigator.aspx
"Класс XPathNavigator в пространстве имен System.Xml.XPath - это абстрактный класс, который определяет модель курсора для навигации и редактирования информационных элементов XML как экземпляров модели данных XQuery 1.0 и XPath 2.0".