根据C#中的某些条件选择xml节点及其“子节点”的值

时间:2016-09-24 14:58:24

标签: c# xml linq xpath

我需要根据另一个节点值满足的条件获取节点值。

这是我正在搜索的xml示例:

<Elements>
<Element>
    <ElementID>A1</ElementID>
    <ElementName>Element A</ElementName>
    <ElementValues>
        <ElementValue>
            <ValueText>A Value</ValueText>
            <ValueDescription>A Type Element</ValueDescription>
        </ElementValue>
    </ElementValues>
</Element>
<Element>
    <ElementID>B1</ElementID>
    <ElementName>Element B</ElementName>
    <ElementValues>
        <ElementValue>
            <ValueText>B Value</ValueText>
            <ValueDescription>B Type Element</ValueDescription>
        </ElementValue>
    </ElementValues>
</Element>
</Elements>

我需要来自具有特定ElementID的节点的ValueText标记的标记值。例如:如果搜索条件为“A1”,则结果应为“A值”。 (例如:如果ElementID =“A1”则获取ValueText.Value ) 请帮我这个(Linq,XPath,无论......),我只是在圈子里旋转而无处可去。

提前致谢!

我真的已经找到了答案,但无法得到类似的答案......

4 个答案:

答案 0 :(得分:1)

如果我是对的,你可以使用XPath表达式

//Element[ElementID/text() = \"B1\"]

它将搜索XML文档中的每个<Element>节点,并选择那些具有<ElementID>子节点的节点,该子节点还具有text()(内部文本)具有的值的属性B1。您可以将“B1”修改为“A1”,您将获得另一个节点,在插入用户输入时要小心,您不希望它们能够修改您的完整XPath表达式,这可能导致漏洞

这是一个示例程序。

    public static void Main(string[] args)
    {
        string xml = @"<Elements>
                    <Element>
                        <ElementID>A1</ElementID>
                        <ElementName>Element A</ElementName>
                        <ElementValues>
                            <ElementValue>
                                <ValueText>A Value</ValueText>
                                <ValueDescription>A Type Element</ValueDescription>
                            </ElementValue>
                        </ElementValues>
                    </Element>
                    <Element>
                        <ElementID>B1</ElementID>
                        <ElementName>Element B</ElementName>
                        <ElementValues>
                            <ElementValue>
                                <ValueText>B Value</ValueText>
                                <ValueDescription>B Type Element</ValueDescription>
                            </ElementValue>
                        </ElementValues>
                    </Element>
                    </Elements>";
        XmlDocument doc = new XmlDocument();
        doc.LoadXml(xml);
        XmlNode root = doc.DocumentElement;

        /* Select all "<Element>" nodes which have an <ElementID> subnode where the text equals "B1". */
        var nodes = root.SelectNodes("//Element[ElementID/text() = \"B1\"]");

        foreach(XmlNode node in nodes){
            Console.WriteLine("Found matching Element: \n {0}", node.InnerXml);
        }
  }          

输出

Found matching Element: 
 <ElementID>B1</ElementID><ElementName>Element B</ElementName><ElementValues><ElementValue><ValueText>B Value</ValueText><ValueDescription>B Type Element</ValueDescription></ElementValue></ElementValues>

答案 1 :(得分:0)

这是一个简单的LINQ查询:

string xml = ...;
var root = XElement.Parse(xml);
string elementID = "A1";
var result = root.Elements("Element")
    .Where(e => e.Element("ElementID").Value == elementID)
    .Select(e => e.Element("ElementValues").Element("ElementValue").Element("ValueText").Value)
    .FirstOrDefault();

答案 2 :(得分:0)

使用XML和LINQ to XML提供了一种使用XML的面向对象方法。例如,当最初解析XML时,整个XML结构是对象,即Root。选择节点时,每个节点都成为它自己的对象。从您的示例中,假设您已经拥有XML是一个字符串(可以是文件或流,无论如何):

var doc = XDocument.Parse(xml);

// retrieve the parent node of A1
XElement a1 = doc.Root.Elements("Element")
                      .First(x => x.Element("ElementID").Value == "A1");

// a1 now has an XML structure of:
// <Element>
//   <ElementID>A1</ElementID>
//   <ElementName>Element A</ElementName>
//   <ElementValues>
//     <ElementValue>
//       <ValueText>A Value</ValueText>
//       <ValueDescription>A Type Element</ValueDescription>
//     </ElementValue>
//   </ElementValues>
// </Element>

// get the value of ValueText w/ a verbose path:
string verbosePath = a1.Element("ElementValues")
                       .Element("ElementValue")
                       .Element("ValueText").Value;

// alternatively, short-circuit the path:
string shortPath = a1.Descendants("ValueText").First().Value;

答案 3 :(得分:0)

使用LINQ完成,将elementId替换为您想要值文本的元素ID。

var xDoc = XDocument.Parse(xml);
var elements = xDoc.Element("Elements");
var element = elements.Elements("Element").Single(e => e.Element("ElementID").Value.Equals(elementId));
var valueText = element.Descendants("ValueText").Single().Value;