程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> .NET網頁編程 >> 關於.NET >> Linq to Xml讀取復雜xml(帶命名空間)

Linq to Xml讀取復雜xml(帶命名空間)

編輯:關於.NET

前言:xml的操作方式有多種,但要論使用頻繁程度,博主用得最多的還是Linq to xml的方式,覺得它使用起來很方便,就用那麼幾個方法就能完成簡單xml的讀寫。之前做的一個項目有一個很變態的需求:C#項目調用不知道是什麼語言寫的一個WebService,然後添加服務引用總是失敗,通過代理的方式動態調用也總是報錯,最後沒辦法,通過發送原始的WebRequest請求直接得到對方返回的一個xml文件。注意過webservice的wsdl文件的朋友應該知道這個是系統生成的xml文件,有點復雜,研究了半天終於能正常讀寫了。今天在這裡和大家分享下。

 

1、介紹之前,首先回顧下Linq to xml的方式讀寫簡單xml的方法

(1)讀取xml

<?xml version="1.0" encoding="utf-8"?>

<BizADsList>

  <adData aid="1" image="baidu.jpg" link="www.baidu.com" title="百度"/>

  <adData aid="2" image="qq.jpg" link="www.qq.com" title="騰訊"/>

</BizADsList>
var strPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, @"data\test.xml");
            XDocument adList = XDocument.Load(strPath);

            var ad = from a in adList.Descendants("BizADsList").Elements("adData")

                     select new

                     {

                         image = a.Attribute("image").Value,

                         link = a.Attribute("link").Value,

                         title = a.Attribute("title").Value

                     };

            string s = "";

            foreach (var a in ad)

                s += a.image;

(2)寫xml

/// <summary>
        /// 根據得到的Document集合生成XML
        /// </summary>
        /// <param name="lstDocumentBD"></param>
        /// <param name="docNE"></param>
        /// <param name="strSpiderTime"></param>
        /// <param name="strNewRate"></param>
        private static void SaveXmlByLstDocument(List<Document> lstDocumentBD, Document docNE, string strSpiderTime, string strNewRate)
        {
            try
            {
                XDocument xDoc = new XDocument();
                XElement xRoot = new XElement(CSpiderConst.XML_ELE_ROOT);

                //1.構造Device節點
                XElement xDevice = new XElement(CSpiderConst.XML_ELE_DEVICE);

                //2.構造NE節點
                XElement xNE = new XElement(CSpiderConst.XML_ELE_NE);
                foreach (var oDocNE in docNE)
                {
                    XElement xItem = new XElement(CSpiderConst.XML_ELE_ITEM, new XAttribute(CSpiderConst.XML_PROP_NAME, oDocNE.Key), oDocNE.Value);
                    xNE.Add(xItem);
                }
                //這裡增加一個<Item name='NewRate'>和<Item name='SpiderTimeEx'>節點用來保存當前這次的利用率和當次的采集時間
                AddNewRateAndSpiderTime(strSpiderTime, strNewRate, xNE);
                xDevice.Add(xNE);

                //3.循環構造BD節點並添加到Device節點中
                foreach (var oDocument in lstDocumentBD)
                {
                    XElement xBD = new XElement(CSpiderConst.XML_ELE_BD);
                    foreach (var oDocBD in oDocument)
                    {
                        XElement xItem = new XElement(CSpiderConst.XML_ELE_ITEM, new XAttribute(CSpiderConst.XML_PROP_NAME, oDocBD.Key), oDocBD.Value);
                        xBD.Add(xItem);
                    }
                    AddNewRateAndSpiderTime(strSpiderTime, strNewRate, xBD);
                    xDevice.Add(xBD);
                }
                xRoot.Add(xDevice);
                xDoc.Add(xRoot);

                //4.保存到采集器本地,以服務器的時間和網元的AssetID來命名
                var strDirectoryPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "ReportFailed\\");
                if (!Directory.Exists(strDirectoryPath))
                {
                    Directory.CreateDirectory(strDirectoryPath);
                }
                xDoc.Save(strDirectoryPath + docNE[TBLDeviceLCBB.PROP_ASSETID] + "_" + strSpiderTime.Replace(":", "_") + ".xml");
            }
            catch
            {
                CLogService.Instance.Debug("保存XML失敗");
            }
        }

通過XDocument、XElement對象和Element()、Elements()兩個方法能完成大部分xml文件的操作。

 

2、進入今天的正題:讀寫帶命名空間的xml文件。

首先來看一段xml

<?xml version="1.0" encoding="utf-8" ?>
<DataSet xmlns="http://WebXml.com.cn/">
  <xs:schema xmlns="" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata" id="getRegion">
    <msdata:aa>
      test
    </msdata:aa>
    <xs:element name="getRegion" msdata:IsDataSet="true" msdata:UseCurrentLocale="true">
      <xs:element name="Province">
        <xs:sequence>
          <xs:element name="RegionID" type="xs:string" minOccurs="0"/>
          <xs:element name="RegionName" type="xs:string" minOccurs="0"/>
        </xs:sequence>
      </xs:element>
    </xs:element>
  </xs:schema>
</DataSet>

第一次看到這個文件確實讓人萌神了,比如需要取一個msdata:IsDataSet="true"這個屬性,該怎麼取...

解析之前,先來分析下這段xml,<DataSet xmlns="http://WebXml.com.cn/">這段裡面有一個xmlns屬性,這個屬性是每一個標簽自帶的屬性,不信你可以新建一個xml文件,然後在任何一個標簽裡面輸入xmlns屬性,後面都會出來很多的系統自帶的命名空間。這個屬性表示所屬標簽在哪個命名空間下面,所以在取的時候要帶上這個命名空間。

先來看看解析的代碼:

var strPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, @"data\test.xml");
var oRoot = XDocument.Load(strPath);
//取DataSet標簽
var oDataSet = oRoot.Element(XName.Get("DataSet", "http://WebXml.com.cn/"));
//取schema標簽
var oSchema = oDataSet.Element(XName.Get("schema", "http://www.w3.org/2001/XMLSchema"));
//取element標簽
var oElement = oSchema.Element(XName.Get("element", "http://www.w3.org/2001/XMLSchema"));//這兩個節點都是以xs打頭,所以命名空間都是xs的命名空間
//取element標簽下面的IsDataSet屬性
var oElementValue = oElement.Attribute(XName.Get("IsDataSet", "urn:schemas-microsoft-com:xml-msdata"));
//取aa標簽
var oAA = oSchema.Element(XName.Get("aa", "urn:schemas-microsoft-com:xml-msdata"));

我們來解析下幾個關鍵的地方:

(1)我們來解析下

<xs:schema xmlns="" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata" id="getRegion">

這一句,最前面的"xs"表示標簽所屬命名空間的變量,xmlns:xs="http://www.w3.org/2001/XMLSchema"這個表示xs這個命名空間的值。所以要得到schema這個標簽需要帶上命名空間var oSchema = oDataSet.Element(XName.Get("schema", "http://www.w3.org/2001/XMLSchema"));這個標簽還定義了另一個命名空間xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"。

(2)再來看看aa標簽

 <msdata:aa>
      test
 </msdata:aa>

msdata就是上面schema標簽裡面定義的另一個命名空間,表示aa標簽屬於msdata命名空間下面。

(3)再看來看屬性的取法:

<xs:element name="getRegion" msdata:IsDataSet="true" msdata:UseCurrentLocale="true">

如果要取msdata:IsDataSet="true",由於這個屬性也帶有命名空間,所以取屬性時也要加上命名空間了。所以需要這樣取。

var oElementValue = oElement.Attribute(XName.Get("IsDataSet", "urn:schemas-microsoft-com:xml-msdata"));

 

      現在大伙們應該對這種xml有一個更加清晰的認識了吧。其實一般情況下這種場景比較少見,因為這麼復雜的xml一般是由引用服務時代理對象去解析的。但如果真的有這麼變態的需求我們也不用擔心了。在此記錄下,以後如果大家遇到希望能節約大伙的時間。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved