С# - получать значения атрибутов из соответствующих узлов XML с использованием запроса XPath
Это не похоже, что это должно быть сложно, но я застрял в настоящее время. Я пытаюсь получить значения атрибутов для определенного атрибута из узлов, которые соответствуют заданной строке запроса XPath. Вот что я до сих пор:
public static IEnumerable<string> GetAttributes(this XmlDocument xml,
string xpathQuery, string attributeName)
{
var doc = new XPathDocument(new XmlNodeReader(xml));
XPathNavigator nav = doc.CreateNavigator();
XPathExpression expr = nav.Compile(xpathQuery);
XPathNodeIterator iterator = nav.Select(expr);
while (iterator.MoveNext())
{
XPathNavigator curNav = iterator.Current;
if (curNav.HasAttributes)
{
XmlNode curNode = ((IHasXmlNode)curNav).GetNode();
if (null != curNode)
{
XmlAttribute attrib = curNode.Attributes[attributeName];
if (null != attrib)
{
yield return attrib.Value;
}
}
}
}
}
В настоящее время это исключение:
System.InvalidCastException: Невозможно передать объект типа "MS.Internal.Xml.Cache.XPathDocumentNavigator" для ввода "System.Xml.IHasXmlNode".
Я собираюсь сделать это неправильно? Есть ли более простой способ получить значения атрибутов от совпадающих узлов?
Ответы
Ответ 1
Для следующего xml:
<root>
<elem att='the value' />
</root>
Вы можете получить текст "значение" с помощью этого кода на С#
XmlDocument xdoc = new XmlDocument();
xdoc.LoadXml(text);
Console.WriteLine(xdoc.SelectSingleNode("/root/elem/@att").Value);
Ответ 2
Если вы используете .net 3.5 или новее, вы можете использовать linq для Xml
Для данного документа xml
<?xml version="1.0" encoding="utf-8" ?>
<root>
<storedProcedures>
<storedProcedure name="usp_GET_HOME_PAGE_DATA">
<resultSet name="Features"/>
<resultSet name="Highlights"/>
</storedProcedure>
<storedProcedure name="usp_GET_FEATURES" />
<storedProcedure name="usp_GET_FEATURE" />
<storedProcedure name="usp_UPDATE_FEATURE" />
<storedProcedure name="usp_GET_FEATURE_FOR_DISPLAY">
<resultSet name="CurrentFeature"/>
<resultSet name="OtherFeatures"/>
</storedProcedure>
<storedProcedure name="usp_GET_HIGHLIGHT_TITLES">
<resultSet name="Highlights"/>
</storedProcedure>
</storedProcedures>
</root>
В следующем выражении linq вы получите значения атрибутов "name" для всех сохраненных процедур node
XDocument xDcoument = XDocument.Load(xmlStoredProcSchemeFile);
var storedProcedureNames = from doc in xDcoument.Descendants("storedProcedure")
select doc.Attribute("name").Value;
вы также можете использовать обычный синтаксис XPath. В приведенном ниже коде переменная node содержит node, идентифицированную именем "usp_GET_HOME_PAGE_DATA", а затем переменная атрибутов содержит все дочерние узлы (атрибуты) выбранного node и это дети.
XmlDocument xmlDocument = new XmlDocument();
xmlDocument.Load(@"C:\inetpub\wwwroot\ASPNETBuilder\BusinessLayer\DataAccessCodeGenerationSchema.xml");
var node = xmlDocument.DocumentElement.SelectSingleNode("./storedProcedures/storedProcedure[@name='usp_GET_HOME_PAGE_DATA']");
var attributes = node.SelectNodes("./resultSet/@name");
Ответ 3
решение исходной задачи исключения исключается...
var doc = new XPathDocument(new XmlNodeReader(xml));
следует заменить на...
var doc = new XmlDocument();
doc.load(*you can either specify the path to the file, the string out of which the xml document is to be generated or specify an xmlreader, look for more overloads*);
Это не вызовет исключения, и код будет работать нормально.