2012-04-26 52 views
2

是否有一种从XML构造类的简单方法。构造的类将用于序列化和反序列化XML。用于序列化的XML类生成器

我有一个定义了很多属性和元素的XML。我是否需要根据该XML手动创建我的课程?或者是有一个实用工具,可用于从XML生成类

感谢,

也先

+0

对于哪种语言? – 2012-04-26 14:54:15

+0

你用什么语言/平台来构建你的班级? – pmartin 2012-04-26 14:54:24

+0

抱歉不清楚。我正在研究Microsoft.NET(VB.NET或C#.NET) – Esen 2012-04-26 15:31:04

回答

1

@Willem van Rumpt:解决方案帮助我创建课堂。但在某些情况下,当我尝试实例化数据集,我最终收到此异常“相同的表不能在两个嵌套的关系中的子表...”

我已经尝试了不同的解决方案,使用xmldocument对象来导航每个节点并生成我的类,可以用来序列化和反序列化XML文件。想到在这里发布它,这将有助于寻找类似解决方案的人。如果有的话,请发布您的优化解决方案。

namespace Utility1 
{ 
public static class XMLHelper 
{ 
    private enum XMLType 
    { 
     Element, 
     Attribute 
    } 
    public static string GenerateXMLClass(string xmlstring) 
    { 
     XmlDocument xd = new XmlDocument(); 
     xd.LoadXml(xmlstring); 
     XmlNode rootNode = xd.DocumentElement; 
     var xmlClassCollection = new Dictionary<string, XMLClass>(); 
     var xmlClass = new XMLClass(); 
     xmlClassCollection.Add(rootNode.Name, xmlClass); 
     CollectAttributes(ref xmlClass, rootNode); 
     CollectElements(ref xmlClass, rootNode); 
     CollectChildClass(ref xmlClassCollection, rootNode); 

     var clsBuilder = new StringBuilder(); 

     clsBuilder.AppendLine("[XmlRoot(\"" + rootNode.Name + "\")]"); 

     foreach (var cls in xmlClassCollection) 
     { 
      clsBuilder.AppendLine("public class " + cls.Key); 
      clsBuilder.AppendLine("{"); 

      foreach (var element in cls.Value.Elements) 
      { 
       if (XMLType.Element == element.XmlType) 
        clsBuilder.AppendLine("[XmlElement(\"" + element.Name + "\")]"); 
       else 
        clsBuilder.AppendLine("[XmlAttribute(\"" + element.Name + "\")]"); 
       clsBuilder.AppendLine("public " + element.Type + element.Name + "{get;set;}"); 
      } 

      clsBuilder.AppendLine("}"); 
     } 

     return clsBuilder.ToString(); 
    } 

    private static void CollectAttributes(ref XMLClass xmlClass, XmlNode node) 
    { 
     if (null != node.Attributes) 
     { 
      foreach (XmlAttribute attr in node.Attributes) 
      { 
       if (null == xmlClass.Elements.SingleOrDefault(o => o.Name == attr.Name)) 
        xmlClass.Elements.Add(new Element("string ", attr.Name, XMLType.Attribute)); 
      } 
     } 
    } 

    private static bool IsEndElement(XmlNode node) 
    { 
     if ((null == node.Attributes || node.Attributes.Count <= 0) && 
        (null == node.ChildNodes || !node.HasChildNodes || (node.ChildNodes.Count == 1 && node.ChildNodes[0].NodeType == XmlNodeType.Text))) 
     { 
      return true; 
     } 
     return false; 
    } 

    private static void CollectElements(ref XMLClass xmlClass, XmlNode node) 
    { 
     foreach (XmlNode childNode in node.ChildNodes) 
     { 
      if (null == xmlClass.Elements.SingleOrDefault(o => o.Name == childNode.Name)) 
      { 
       var occurance = node.ChildNodes.Cast<XmlNode>().Where(o => o.Name == childNode.Name).Count(); 
       var appender = " "; 
       if (occurance > 1) 
        appender = "[] "; 

       if(IsEndElement(childNode)) 
       { 
        xmlClass.Elements.Add(new Element("string" + appender, childNode.Name, XMLType.Element)); 
       } 
       else 
       { 
        xmlClass.Elements.Add(new Element(childNode.Name + appender, childNode.Name, XMLType.Element)); 
       } 
      } 
     } 
    } 

    private static void CollectChildClass(ref Dictionary<string, XMLClass> xmlClsCollection, XmlNode node) 
    { 
     foreach (XmlNode childNode in node.ChildNodes) 
     { 
      if (!IsEndElement(childNode)) 
      { 
       XMLClass xmlClass; 
       if (xmlClsCollection.ContainsKey(childNode.Name)) 
        xmlClass = xmlClsCollection[childNode.Name]; 
       else 
       { 
        xmlClass = new XMLClass(); 
        xmlClsCollection.Add(childNode.Name, xmlClass); 
       } 
       CollectAttributes(ref xmlClass, childNode); 
       CollectElements(ref xmlClass, childNode); 
       CollectChildClass(ref xmlClsCollection, childNode); 
      } 
     } 
    } 

    private class XMLClass 
    { 
     public XMLClass() 
     { 
      Elements = new List<Element>(); 
     } 
     public List<Element> Elements { get; set; } 
    } 

    private class Element 
    { 
     public Element(string type, string name, XMLType xmltype) 
     { 
      Type = type; 
      Name = name; 
      XmlType = xmltype; 
     } 
     public XMLType XmlType { get; set; } 
     public string Name { get; set; } 
     public string Type { get; set; } 
    } 
    } 
} 

感谢,

也先

+0

顺便说一句,这将创建具有字符串属性的类。需要验证该值并分配适当的属性类型。 – Esen 2012-04-27 15:22:20

+0

我不得不说,我个人会避免自己解决这个问题(鉴于许多(很多)陷阱);如果它适合你:伟大:) – 2012-04-27 15:57:36

2

有一个关于倒过来: 使用xsd.exe,你可以先创建XML文件架构(XSD) ,然后可以将其用作xsd.exe的输入以从模式生成类。 即(从命令提示):

xsd.exe myXmlFile.xml 

输出myXmlFile.xsd

和下

xsd.exe myXmlFile.xsd 

以产生从XSD文件类。

+0

这不提供我正在寻找的直接结果。但这个提示很有帮助。我可以从那里生成数据集,然后可以生成我正在寻找的类。感谢您的帮助Rumpt。 – Esen 2012-04-26 18:39:27

3

而且对威廉的帖子:

这将生成XSD(没有数据集

xsd.exe myCustom.xml 

这将生成C#类:

xsd.exe myCustom.xsd /c 
+0

什么是生成的C#类?这不是一个数据集?我没有让你code4life – Esen 2012-04-27 20:54:21