XSD文件生成C#VO實體類

来源:http://www.cnblogs.com/ZXdeveloper/archive/2016/03/06/5248657.html
-Advertisement-
Play Games

最近公司要做一個項目,需要和現有的其他項目對接,由於不知道他們的資料庫,只有XSD文件。所以,我們在修改相應的程式時,就需要根據他們提供的XSD文件,來寫我們的VO實體類,由於我寫過根據Oracle資料庫生成VO實體類,因此這次的這個活也就很自然的落在了我的頭上。 一、XSD 首先什麼是XSD,我就


  最近公司要做一個項目,需要和現有的其他項目對接,由於不知道他們的資料庫,只有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 - 4, 4).Equals("Grid") || node.Type.Substring(node.Type.Length - 4, 4).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 - 4, 4).Equals("Grid") || node.Name.Substring(node.Name.Length - 4, 4).Equals("List")))
            {
                foreach (var root in VoList)
                {
                    if (root.Name.Equals(node.Name))
                    {
                        node.Type = root.Children.First().Name;
                    }
                }
            }
            else
            {
                Console.WriteLine("未能轉化的內容" + node.Name);
            }
        }
View Code

  在替換的過程中,會有一種情況,即一個實體類中包含List<>泛型集合,我相信大家在開發過程中,見過這種情況,因此,在替換的時候,需要判斷一下,通過我們的分析,一般這種的欄位type是在基礎類中找不到的,而是在當前的文件中進行查找,因此,當遇到的時候,還需要再次迴圈一遍自身集合,從而找到實體集合,將他的Name屬性賦值給當前的Type欄位。

四、生成CS文件

  最後一步,就是生成CS文件了,和我上一篇文章大同小異,不過有一點需要註意,因為資料庫生成的實體類是沒有List類型的,但是這個XSD生成的有,因此需要稍加變動一下生成方法。

註:以上解析方法,同樣適用於下圖的一般request和response文件,特殊的需要另當處理。

 五、Demo效果圖

一個XSD生成了多個VO文件,每一個VO就是一個實體類

 

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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 系統來自系統媽:http://www.xitongma.com 系統概述 電腦公司ghost win7 x86(32位)萬能裝機版集成的軟體符合電腦公司及電腦城裝機絕大多數人要求及喜好,既大眾,又時尚,人人喜歡,處處適用。自動判斷筆記本電腦或台式電腦鍵盤類型,並自動開啟台式鍵盤數字指示燈,筆記本鍵盤
  • 開源監控軟體對比 Cacti(英文含義仙人掌) 是一套基於PHP、MySQL、SNMP和RRDtool開發的網路流量監測圖形分析工具,它通過snmpget來獲取數據使用RRDtool繪圖,簡化RRDtool使用。提供了非常強大的數據和用戶管理功能,可以指定每一個用戶能查看樹狀結構、主機設備以及任何一...
  • 最近在移植QQ物聯協議,需要讀取設備mac地址。 讀取CC2541設備MAC地址的方法,有直接讀取mac寄存器,也可以調用協議棧API。這裡採用的是調用API的形式。 1 /*利用協議棧API讀取MAC地址*/ 2 static uint8 macaddr[B_ADDR_LEN]={0}; // m
  • 2. 3 4 5 6 7 QQ 463431476 8 9
  • 前兩天弄了個自動配置JDK環境變數的小工具(詳情:http://www.cnblogs.com/chr-wonder/p/5208541.html)。在調試過程中發現了一些小問題。在此做以記錄。 在寫入Path環境變數過程中,由於我是在Path變數的末尾添加新項的,所以為了避免和原有的項衝突(或者說
  • 1、介面: 介面與抽象類一樣,也是表示某種規則,一旦使用了該規則,就必須實現相關的方法。對於C#語言而言,由於只能繼承自一個父類,因此若有多個規則需要實現,則使用介面是個比較好的做法。 2、介面的定義 interface 介面名 { 方法聲明; } 3、不同介面中若有多個相同名稱的方法,則需要顯式指
  • 我想大家對DateTime.ToString()方法的用法肯定已經非常熟悉了,但我想大家用過的大部分用法都是:DateTime.ToString(“format”),不過本文想講述的是它的另一個重載方法DateTime.ToString("format",IFormatProvider)。 如果大家
  • 文本框賦值時,設置其格式非常方便,將值的格式設定好即可。例如: Dim i as integer = 4333 me.textbox1.text = String.Format("{0:00.0}", i) 但對於文本框綁定到數據源時,此方式無效。可以採取下列方法: 在Form的Load事件中,或其
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...