使用c#中的标记内的属性生成XML

时间:2014-01-05 07:37:10

标签: c# xml sqlite

我有以下XML结构。

<a>
  <b>
    <c>
      <d>54</d>
      <e>US</e>
      <f>Canada</f>          
      <g>
        <h>
          <i1>39</i1>              
          <i2>Belgium</i2>
          <i3>Russia</i3>
        </h>
        <h>
          <i1>43</i1>              
          <i2>Fran</i2>
          <i3>Ger</i3>
        </h>
      </g>
    </c>
    <c>
      <d>5</d>
      <e>US</e>
      <f>Can</f>          
      <g>
        <h>
          <i1>29</i1>              
          <i2>Brit</i2>
          <i3>Ice</i3>
        </h>
        <h>
          <i1>95</i1>              
          <i2>Russia</i2>
          <i3>Nero</i3>
        </h>
        <h>
          <i1>4323</i1>              
          <i2>Polska</i2>
          <i3>503</i3>
        </h>
      </g>
    </c>         
  </b>
  <b2>
    <c2>
      <d2>551</d2>
      <d3>Indo</d3>
      <e2>
        <f2>
          <g2>Irnna</g2>
          <g3>Mehak</g3>
          <g4>Vodka</g4>              
          <h2>
            <i2>
              <j1>44</j1>                  
              <j6>Germ0</j6>                  
            </i2>
            <i21>Finish</i21>
        </f2>    
      </e2>  
      <f3>544</f3>
      <g3>fur</g3>
      <h3>denmark</h3>
     <k1>lur</k1>
     <k2>Bern</k2>
     <k3>Zurick</k3>
     <k4>Italy</k4>
    </c2>
  </b2>
 <b3>35</b3>
 <b4>ferha</b4>
 <b5>english</b5>
 <b6>334</b6>
</a>

我需要将其转换为以下内容 -

<a>
      <b>
        <c d="54"; e="US"; f="Canada">        
         <g>
            <h i1="39"; i2 = "Belgium"; i3 ="Russia">               
            <h i1="393"; i2 = "Fran"; i3 ="Ger">                
         </g>
        </c>
        <c>
       .
       .
     .
    .

表示将所有属性放在标记中。我使用了以下c#代码 -

var xDoc = XDocument.Load("my.xml");
                var xNode =
                    new XElement("a",
                        from name in xDoc.Root.Elements()
                        select new XElement(name.Name,
                            from names in name.Elements()
                            select new XElement(names.Name,
                                from namesElement in names.Elements()
                                where namesElement.Name.LocalName != "g"
                                select new XAttribute(namesElement.Name.LocalName, (string)namesElement),
                                new XElement("g",
                                    from Parts in names.Element("g").Elements()
                                    select new XElement(Parts.Name,
                                        from routeElement in Parts.Elements()
                                        select new XAttribute(routeElement.Name.LocalName, (string)routeElement)))))); 

但是,我收到了错误。能不能给我任何提示或纠正我的代码中的错误?实际上,它一直工作到(上半年)。当结构在后半部分发生变化时,它会显示“对象引用未设置为对象实例”,“重复属性”等错误。你能告诉我我的方法是否正确吗?逻辑结构有什么问题吗?

1 个答案:

答案 0 :(得分:0)

您的示例输入XML文档缺少h2元素的结束标记。此外,输入XML文档中的值与预期输出XML中的值不匹配。话虽如此,我使用了您的示例输入(我添加了缺少的h2结束标记)并通过以下代码运行它:

XDocument xDoc = XDocument.Load("my.xml");
XElement[] elements = xDoc.DescendantNodes()
    .Where(n => n.NodeType == XmlNodeType.Text).Select(n => n.Parent).ToArray();

foreach (XElement element in elements)
{
    element.Parent.SetAttributeValue(element.Name, element.Value);
    element.Remove();
}

结果输出是:

<a b3="35" b4="ferha" b5="english" b6="334">
  <b>
    <c d="54" e="US" f="Canada">
      <g>
        <h i1="39" i2="Belgium" i3="Russia" />
        <h i1="43" i2="Fran" i3="Ger" />
      </g>
    </c>
    <c d="5" e="US" f="Can">
      <g>
        <h i1="29" i2="Brit" i3="Ice" />
        <h i1="95" i2="Russia" i3="Nero" />
        <h i1="4323" i2="Polska" i3="503" />
      </g>
    </c>
  </b>
  <b2>
    <c2 d2="551" d3="Indo" f3="544" g3="fur" h3="denmark" k1="lur" k2="Bern" k3="Zurick" k4="Italy">
      <e2>
        <f2 g2="Irnna" g3="Mehak" g4="Vodka">
          <h2 i21="Finish">
            <i2 j1="44" j6="Germ0" />
          </h2>
        </f2>
      </e2>
    </c2>
  </b2>
</a>

这看起来像你正在寻找的但如果没有,也许代码可以指出你正确的方向。

我使用了LINQ方法语法而不是您正在使用的查询语法,但我不认为该部分很重要。但是我注意到你正在寻找特定的元素,比如“a”或“g”,我想看看是否可以用更一般的方式来完成。

我写这个时也遇到了空引用异常。例如,我最初将elements变量作为IEnumerable<XElement>,但稍后我会从foreach中的父项中删除一个元素,并且在下一次迭代中会有一些LINQ延迟执行所以它会返回到DescendantNodes(),但它不能正常删除元素,我会看到空引用异常。因此,不是在LINQ下改变事物,而是ToArray()集合,以便所有执行立即发生,而我使用数组。我提到这一切都是因为也许这种事情也在发生在你身上。