最近公司要做一个项目,需要和现有的其他项目对接,由于不知道他们的数据库,只有XSD文件。所以,我们在修改相应的程序时,就需要根据他们提供的XSD文件,来写我们的VO实体类,由于我写过根据Oracle数据库生成VO实体类,因此这次的这个活也就很自然的落在了我的头上。

一、XSD

  首先什么是XSD,我就不解释了,因为我也不是很清楚,第一次接触,具体的详解度娘一下很多,XSD是XML Schema Definition 的简称,既然是XML,那读取方式,就按照XML的方式就可以了。

二、解析的XSD文件

  我要解析的XSD文件是如下这个样子的,大家可以看到这个这个文件还需要加载两个xsd文件,但是通过我们的分析,对于我们这个项目来说,如果要生成实体类,除了需要解析下面这个图示XSD以外,还需要解析一个type类型xsd,也就是图中include的第二个xsd文件。

要解析的XSD

  先来看看上面这个VO实体类文件,通过截图,可以看出来,这个是一个VO实体类包含了好多个字段,但是每个字段的type又没有表示出来,因此需要到下面这个截图的xsd里去寻找,因为下图中的name对应着上图中的type,而下图中的base正是我们需要的字段类型。

<xs:simpleType name="acbfyhdcbfyze">
<xs:annotation>
<xs:documentation>按成本费用核定成本费用总额</xs:documentation>
</xs:annotation>
<xs:restriction base="xs:double"/>
</xs:simpleType>
<xs:simpleType name="acbfyhdhdlrl">
<xs:annotation>
<xs:documentation>按成本费用核定核定的利润率</xs:documentation>
</xs:annotation>
<xs:restriction base="xs:double"/>
</xs:simpleType>
<xs:simpleType name="acbfyhdhsdsre">
<xs:annotation>
<xs:documentation>按成本费用核定换算的收入额</xs:documentation>
</xs:annotation>
<xs:restriction base="xs:double"/>
</xs:simpleType>
<xs:simpleType name="acbfyhdynssde">
<xs:annotation>
<xs:documentation>按成本费用核定应纳税所得额</xs:documentation>
</xs:annotation>
<xs:restriction base="xs:double"/>
</xs:simpleType>

基础XSD

三、分析和解析

  通过分析上面的两个文件,由于要生成实体类文件,因此要解析的文件需要读取成功以后,生成一个List列表,而基础XSD则应该解析成一个Dictionary,而这个Dictionary中只需要包含name和base,然后通过循环List列表,依次进Dictionary中进行相应的替换,就可以得到字段的类型,从而生成一个最终需要的List集合。

有了分析以后,那就是付诸实践了,首先要能解析XSD文件,解析XSD和解析XML差不多,就是通过获取节点集合,然后循环得到自己想要的东西。

  1、根据节点设置静态字段,方便在解析的时候进行调用

public static class ConstField
{
public const string Schema = "xs:schema";
public const string ComplexType = "xs:complexType";
public const string Element = "xs:element";
public const string Annotation = "xs:annotation";
public const string Documentation = "xs:documentation";
public const string Sequence = "xs:sequence";
public const string SimpleType = "xs:simpleType";
public const string Restriction = "xs:restriction";
}

静态字段

  2、写一个实体类,其中包含name(字段名),type(字段类型),note(注释),为了生存List时用

public class AnalysisVo
{
private string name;
private string note;
private string type;
private List<AnalysisVo> children;
/// <summary>
/// 名称
/// </summary>
public string Name
{
get
{
return name;
} set
{
name = value;
}
}
/// <summary>
/// 备注
/// </summary>
public string Note
{
get
{
return note;
} set
{
note = value;
}
}
/// <summary>
/// 类型
/// </summary>
public string Type
{
get
{
return type;
} set
{
type = value;
}
}
/// <summary>
/// 子集
/// </summary>
public List<AnalysisVo> Children
{
get
{
return children;
} set
{
children = value;
}
}
}

实体类

  3、解析XSD文件

public void Analysis()
{
XmlElement rootElement = _document.DocumentElement;//获取根节点
XmlNodeList complexTypeNodes = rootElement.GetElementsByTagName(ConstField.ComplexType);//获取complexType子节点集合
foreach (XmlNode complexTypeNode in complexTypeNodes)
{
AnalysisVo vo = new AnalysisVo(); XmlElement voElement = (XmlElement)complexTypeNode;
string nameVo = voElement.GetAttribute("name");//获取类的名字
string noteVo = voElement.FirstChild.InnerText;//获取类的注释 List<AnalysisVo> childrenList = new List<AnalysisVo>(); XmlNodeList elementNodes = voElement.GetElementsByTagName(ConstField.Element);//获取element子节点集合
foreach (XmlNode elementNode in elementNodes)
{
AnalysisVo childVo = new AnalysisVo(); XmlElement fieldElement = (XmlElement)elementNode;
string nameField = fieldElement.GetAttribute("name");//获取字段名字
string typeField = fieldElement.GetAttribute("type");//获取字段类型
string noteField = ""; //获取字段注释
if(fieldElement.FirstChild!=null)
noteField= fieldElement.FirstChild.InnerText;
childVo.Name = nameField;
childVo.Type = typeField;
childVo.Note = noteField; childrenList.Add(childVo);
} vo.Name = nameVo;
vo.Note = noteVo;
vo.Children = childrenList; VoList.Add(vo);
}
}

解析实体类XSD文件

public void Analysis()
{
XmlElement rootElement = _document.DocumentElement;//获取根结点
XmlNodeList simpleTypeNodes = rootElement.GetElementsByTagName(ConstField.SimpleType);//获取simpleType子节点集合
foreach(XmlNode simpleTypeNode in simpleTypeNodes)
{
XmlElement dicElement = (XmlElement)simpleTypeNode;
string dicKey = dicElement.GetAttribute("name");//获取类型名字
string dicValue = ((XmlElement)dicElement.LastChild).GetAttribute("base");//获取类型内容
dicValue = dicValue.Replace("xs:", "");
DicType.Add(dicKey, dicValue);
}
}

解析Type类型XSD

  4、将以上得到的两个文件进行整合,从而生成一个最终的List文件

private void ReplaceContent(AnalysisVo node)
{
//正常在基础的字段表中存在
if (DicType.ContainsKey(node.Type))
{
node.Type = DicType[node.Type];
}//要解析的里面没有type字段,只有name字段 yanc 20160304
else if (string.IsNullOrEmpty(node.Type) && DicType.ContainsKey(node.Name))
{
node.Type = DicType[node.Name];
}
else//List类型的没有存储在基础的字段表中,需要循环遍历本身的List列表
if(!string.IsNullOrEmpty(node.Type)&&(node.Type.Substring(node.Type.Length - , ).Equals("Grid") || node.Type.Substring(node.Type.Length - , ).Equals("List")))
{
foreach (var root in VoList)
{
if (root.Name.Equals(node.Type))
{
node.Type = root.Children.First().Type;
}
}
}//要解析的里面没有type字段,只有name字段 yanc 20160304
else if(string.IsNullOrEmpty(node.Type) && (node.Name.Substring(node.Name.Length - , ).Equals("Grid") || node.Name.Substring(node.Name.Length - , ).Equals("List")))
{
foreach (var root in VoList)
{
if (root.Name.Equals(node.Name))
{
node.Type = root.Children.First().Name;
}
}
}
else
{
Console.WriteLine("未能转化的内容" + node.Name);
}
}

  在替换的过程中,会有一种情况,即一个实体类中包含List<>泛型集合,我相信大家在开发过程中,见过这种情况,因此,在替换的时候,需要判断一下,通过我们的分析,一般这种的字段type是在基础类中找不到的,而是在当前的文件中进行查找,因此,当遇到的时候,还需要再次循环一遍自身集合,从而找到实体集合,将他的Name属性赋值给当前的Type字段。

四、生成CS文件

  最后一步,就是生成CS文件了,和我上一篇文章大同小异,不过有一点需要注意,因为数据库生成的实体类是没有List类型的,但是这个XSD生成的有,因此需要稍加变动一下生成方法。

注:以上解析方法,同样适用于下图的一般request和response文件,特殊的需要另当处理。

五、Demo效果图

一个XSD生成了多个VO文件,每一个VO就是一个实体类

代码在GitHub上,希望大家一起帮忙修改,毕竟foreach嵌套foreach效率不是很高。

XSD文件生成C#VO实体类的更多相关文章

  1. (转) 使用jdk的xjc命令由schema文件生成相应的实体类

    背景:在webservice的开发过程中涉及到这一知识点,又必要来学习一下. 1 根据编写的schema来生成对应的java实体 1.1 实战 xcj命令有schema文件生成Java实体类 1.使用 ...

  2. 根据xsd文件生成对应的C#类,然后创建对应的xml文件

    首先用xsd文件生产对应的C#类,这个VS已经自带此工单,方法如下: 1. 打开交叉命令行工具 2. 输入如下指令 xsd d:\123.xsd /c /language:C# /outputdir: ...

  3. 如何由XSD自动生成XML和实体类

    项目中有时候要用XML作为数据源,因此需要定义XML文件和相应的类,最佳方法是首先定义XSD,然后自动生成实体类,最后生成XML和填充数据:读取XML数据源的时候,首先用XSD验证XML数据格式,然后 ...

  4. 用vs2012的命令利用xsd文件生成对应的C#类,把xml的string类型映射到生成的类

    输入命令: xsd d:\TDDOWNLOAD\atom-author-link.xsd /c /language:C# /outputdir:d:\ 含义: 将d:\TDDOWNLOAD\atom- ...

  5. 使用jdk的xjc命令由schema文件生成相应的实体类

    xjc D:\operate-process.xsd -d D:\workspace\wmsc\src\main\java -p com.yd.wmsc.util operate-process.xs ...

  6. 5.7 Liquibase:与具体数据库独立的追踪、管理和应用数据库Scheme变化的工具。-mybatis-generator将数据库表反向生成对应的实体类及基于mybatis的mapper接口和xml映射文件(类似代码生成器)

    一. liquibase 使用说明 功能概述:通过xml文件规范化维护数据库表结构及初始化数据. 1.配置不同环境下的数据库信息 (1)创建不同环境的数据库. (2)在resource/liquiba ...

  7. 使用T4模板生成MySql数据库实体类

    注:本文系作者原创,但可随意转载. 现在呆的公司使用的数据库几乎都是MySQL.编程方式DatabaseFirst.即先写数据库设计,表设计按照规范好的文档写进EXCEL里,然后用公司的宏,生成建表脚 ...

  8. c#实例化继承类,必须对被继承类的程序集做引用 .net core Redis分布式缓存客户端实现逻辑分析及示例demo 数据库笔记之索引和事务 centos 7下安装python 3.6笔记 你大波哥~ C#开源框架(转载) JSON C# Class Generator ---由json字符串生成C#实体类的工具

    c#实例化继承类,必须对被继承类的程序集做引用   0x00 问题 类型“Model.NewModel”在未被引用的程序集中定义.必须添加对程序集“Model, Version=1.0.0.0, Cu ...

  9. Myeclipse 10使用hibernate生成注解(annotation)实体类

    以MySQL数据库为例,请在数据库里面建好对应的表. 1.配置数据库链接 打开Myelipse Database Explorer视图 Window-->Open Perspective--&g ...

随机推荐

  1. CC countari & 分块+FFT

    题意: 求一个序列中顺序的长度为3的等差数列. SOL: 对于这种计数问题都是用个数的卷积来进行统计.然而对于这个题有顺序的限制,不好直接统计,于是竟然可以分块?惊为天人... 考虑分块以后的序列: ...

  2. 【BZOJ】3997: [TJOI2015]组合数学

    题意 \(N \times M\)的网格,一开始在\((1, 1)\)每次可以向下和向右走,每经过一个有数字的点最多能将数字减1,最终走到\((N, M)\).问至少要走多少次才能将数字全部变为\(0 ...

  3. 【BZOJ1857】[Scoi2010]传送带 三分法

    三分套三分,挺神奇的...每次找到,每个传送带的上下两个三等分点,下面那个小,则一定有更优的在中间. #include <iostream> #include <cstdio> ...

  4. css 简析folat

    1.float?? 不知道大家是否还记得之前我们讲过页面是文档流,具体什么是文档流,我就不说了?于是我们页面布局如果用div的话,那么块状的元素是怎么排列的,什么叫块状自己去看? 如果我们呢用div布 ...

  5. Thinkphp3.2.3路径书写注意

    尽量不要这样写: ./public/img/a.jpg 应该这样写:__PUBLIC__/img/a.jpg 不然会引起不兼容  如首页地址 http://192.168.1.100/rjshop/时

  6. 解决python编码格式错误问题

    一:前言 遇到问题:print输入汉字时提示错误信息 UnicodeDecodeError: 'ascii' codec can't decode byte 0x?? in position 1: o ...

  7. nodejs的实现原理和搭建服务器(动态)

    心得体会    今天是我学习的Node.js的第二天,所谓的node.js其实它是javascript编写的服务器的语言,同时它又是属于后台的框架,是一个开放性的平台. 一.相关理论知识: 我们可以用 ...

  8. 异步调用window.open时被浏览器阻止新窗口解决方案

    var wyWindow = window.open('_blank');$http.post($rootScope.baseUrl + '/Interface0231A.ashx', { userF ...

  9. 使用VS2013进行单元测试

    这次的作业安装了VS2013,对于它的安装过程我就不再细说了,归结起来就是一个字——等,尤其是语言包,最后只好放弃了装语言包,凭借我3级半的英语水平,明白这些没有问题——这仅仅个玩笑话,其实我是用有道 ...

  10. 一图搞定【实战Java高并发程序设计】

    来了解下java并发的技术点吧.这里面包括了并发级别.算法.定律,还有开发包.在过去单核CPU时代,单任务在一个时间点只能执行单一程序,随着多核CPU的发展,并行程序开发就显得尤为重要.这本书主要介绍 ...