1

我正在尝试将 XML 数据转换为字典。我遇到了相同节点名称的问题。C# .Net 3.5

示例 XML = 问题是我无法控制它。我只需要处理它。

<?xml version="1.0" encoding="utf-8"?>
<Root>
  <a1>val1</a1>
  <a2>val2</a2>
  <Parameter>
    <ParameterName>param1</ParameterName>
    <ParameterValue>paramval1</ParameterValue>
  </Parameter>
  <Parameter>
    <ParameterName>param2</ParameterName>
    <ParameterValue>paramval2</ParameterValue>
  </Parameter>
</Root>

我的尝试:

XMLStream.Position = 0;
        XElement xmlDetails2 = XElement.Load(new System.IO.StreamReader(XMLStream));
        var x = xmlDetails2.Elements().ToDictionary(
            e => e.Name.LocalName,
            e => e.Elements()
                      .ToDictionary(
                          f => f.Name.LocalName,
                          f => f.Value));

我得到的错误(这当然是有道理的):

An item with the same key has already been added.

预期结果(来自示例 xml):

a1 => val1
a2 => val2
param1 => paramval1
param2 => paramval2
...

我根据@LB 的建议创建了自己的。这不是最好的解决方案,但它现在有效。

public void XMLTODictionary(XElement xmlDetails, ref Dictionary<string, string> dic)
{
    foreach (var node in xmlDetails.Elements())
    {
        if (node.Name.LocalName.Equals("parameter", StringComparison.CurrentCultureIgnoreCase))
        {
                dic.Add(node.Element("ParameterName").Value, node.Element("ParameterValue").Value);
        }
        else
        {
            dic.Add(node.Name.LocalName, node.Value);
        }
    }
}
4

4 回答 4

0

除非您提供唯一的、不可为空的键,否则您将无法导入字典结构。我能想到的几个选择:

  • 在导入期间创建人工密钥
  • 请改用 ToLookup() 扩展方法。这不会返回字典,但可能适合您的需求。更好地了解您为什么需要字典,我可以提供更多帮助。
于 2012-12-06T19:53:44.630 回答
0

在这里使用 DynamicXml怎么样

dynamic root = DynamicXml.Load("a.xml");

Console.WriteLine(root.a1);
foreach (var p in root.Parameter)
{
    Console.WriteLine("{0}:{1}",p.ParameterName, p.ParameterValue);
}
Console.WriteLine(root.Parameter[0].ParameterValue);

编辑

一种通用的方法是获取字典Dictionary<string,object> 但是将 xml 转换为字典时存在一些问题。例如

<a1>val1</a1>

dict["a1"]会返回val1,但是这个 xml 会返回什么

<a1 name="valAttr"><name>valName</name><a1>

dict["a1"]["name"]? valAttr还是valName

dict["a1"]考虑到您的示例,和之间的唯一区别dict["Parameter"]Parameter在同一个父级下不止一次存在,它应该被视为一个数组而不是单个元素。

DynamicXml试图解决这些问题。当然还有很大的改进空间,但它应该可以满足基本需求。

于 2012-12-06T20:01:06.750 回答
0

这是一种将数据提炼成元组然后输出字典的方法。请注意,它不会检查重复键,但可以添加:

string data = @"<?xml version=""1.0"" encoding=""utf-8""?>
<Root>
  <a1>val1</a1>
  <a2>val2</a2>
  <Parameter>
    <ParameterName>param1</ParameterName>
    <ParameterValue>paramval1</ParameterValue>
  </Parameter>
  <Parameter>
    <ParameterName>param2</ParameterName>
    <ParameterValue>paramval2</ParameterValue>
  </Parameter>
</Root>";

var elements = XDocument.Parse( data )
                        .Element("Root")
                        .Descendants();

var asTupleChildren =  elements.Where (e => e.HasElements)
                               .Select (e => new Tuple<string,string>(e.Element("ParameterName").Value, e.Element("ParameterValue").Value ));

var asTupleElements =  elements.Where (e => e.HasElements == false)
                               .Where (e => e.Name != "ParameterName" && e.Name != "ParameterValue" )
                               .Select (e => new Tuple<string,string>(e.Name.ToString(), e.Value ));


var asDictionary =  asTupleElements.Concat(asTupleChildren)
                                   .ToDictionary (te => te.Item1, te => te.Item2);


/* asDictionary is a Dictionary<String,String> (4 items)

a1 val1
a2 val2
param1 paramval1
param2 paramval2

*/
于 2012-12-06T20:51:21.897 回答
0

刚做了这个,对我有用。有点笨重。将其用作模板。

        public static Dictionary<string, dynamic> RecDis(string ThisXML)
    {
        Dictionary<string, dynamic> ThisBlock = new Dictionary<string, dynamic>();

        XElement doc = XElement.Parse(ThisXML);

        XNode[] ThisNoideArray = doc.Nodes().ToArray();

        foreach (XNode park in ThisNoideArray)
        {
            XElement parker = XElement.Parse(park.ToString());

            if (parker.HasElements)
            {
                ThisBlock.Add(parker.Name.ToString(), RecDis(parker.ToString()));
            }
            else
            {
                ThisBlock.Add(parker.Name.ToString(), parker.Value.ToString());
            }
        }

        return ThisBlock;
    }
于 2016-04-04T13:41:13.280 回答