內容提要
1.解析Xml文件有哪些方法?各有什麼優缺點?
2.如何用XPath解析xml文檔的要點。
先來看看解析xml文件的方法都有哪些吧,本段文字來自網絡,可以幫助大家對這個問題有個概要的瞭解。
在程序中訪問並操作XML文件一般有兩種模型:流模型和DOM(文檔對象模型)。流模型中有兩種變體——“推”模型和“拉”模型。
“推”模型也就是常說的SAX,SAX是一種靠事件驅動的模型。它每發現一個節點就用“推”模型引發一個事件,而我們必須編寫這些事件的處理程序,很麻煩。
.NET中使用的是基於“拉”模型的實現方案。 “拉”模型在遍歷文檔時會把感興趣的文檔部分從讀取器中拉出,不需要引發事件,允許我們以編程的方式訪問文檔,這大大的提高了靈活性,“拉”模型可以選擇性的處理節點。在.NET中,“拉”模型通過XML閱讀器(XMLTextReader類)來實現的。該類提供Xml文件讀取的功能,它可以驗證文檔是否格式良好,如果不是格式良好的Xml文檔,該類在讀取過程中將會拋出XmlException異常。任何時候在內存中只有當前節點,但它是隻讀的,向前的,不能在文檔中執行向後導航操作。
DOM的好處在於它允許編輯和更新XML文檔,可以隨機訪問文檔中的數據,可以使用XPath查詢。但是,DOM的缺點在於它需要一次性的加載整個文檔到內存中,對於大型的文檔,這會造成資源問題。在.NET中使用XML DOM分析器(XMLDocument)實現DOM模型。
因此,.NET Framework完全支持XML DOM模式,但它不支持SAX模式。.NET Framework支持兩種不同的分析模式:XML DOM分析器(XMLDocument類)和XML閱讀器(XMLTextReader類),不支持SAX分析器,但這並不意味着它沒有提供類似SAX分析器的功能。通過XML閱讀器可以將SAX的所有的功能很容易的實現及更有效的運用。
在項目中,我們選用xpath的方式來解析xml文檔。這是基於以下的幾點原因:
1, 文件大小。要處理的文件不大,一般都在幾百K到1M。
2, XPath的靈活性。不需要獲取文檔的全部數據,只需要獲取大部分想要的數據。
3, 學習代價低。符合一般的思維習慣,通過Path獲取結果。
通過XPath的方式解析xml文檔,需要先加載文檔,然後再讀取想要的節點值。
xml文檔
protected XmlDocument doc = null;
xml文檔的根元素(節點)
protected XmlElement root = null;
xml文檔的名空間管理器
protected XmlNamespaceManager nsmgr = null;
接下來就是加載文檔了
protected void LoadXmlFile(FileInfo xmlFile)
{
if (xmlFile == null || !xmlFile.Exists)
{
throw new FileNotFoundException(string.Format("要解析的文件不存在{0}。",xmlFile.FullName));
}
//加載文件
this.doc = new XmlDocument();
doc.Load(xmlFile.FullName);
//準備讀取文件
root = doc.DocumentElement;
string nameSpace = root.NamespaceURI;
nsmgr = new XmlNamespaceManager(doc.NameTable);
nsmgr.AddNamespace("ns", nameSpace);
}
這裏有幾行要注意。
這兩行是取得xml文檔的名空間
string nameSpace = root.NamespaceURI;
這兩行是建立xml文檔的名空間管理器
nsmgr.AddNamespace("ns", nameSpace);
如果你的xml文檔有名空間,則這部分的代碼是必不可少的。
接下來就是讀取文檔節點的值了
這裏兩個傳入參數prefixPath是節點的上級節點路徑,xRelativePath是要讀取的節點名稱。
另外,變量XmlFileInfo是要加載的xml文件。
protected string GetNodeValue(string prefixPath, string xRelativePath)
{
if (doc == null)
{
LoadXmlFile(XmlFileInfo);
}
string xPath = string.Empty;
if (!string.IsNullOrEmpty(xRelativePath))
{
if (!string.IsNullOrEmpty(prefixPath))
{
xPath = prefixPath + xRelativePath;
}
else
{
xPath = xRelativePath;
}
}
xPath = xPath.Replace("/", "/ns:");
XmlNode node = root.SelectSingleNode(xPath, nsmgr);
if (node == null)
{
return null;
}
return node.InnerXml;
}
可能有的朋友要問,爲什麼要設置兩個參數prefixPath和xRelativePath呢,其實這個沒有多大的關係,我只是爲了自己覺得方便,你也可以在方法外確定了這個XPath,在方法中只設置一個傳入參數,效果是一樣的。
注意這一行:
xPath = xPath.Replace("/", "/ns:");
如果你的xml文檔帶名空間,則這行是比不可少的,否則會出現找不到節點,無法解析的情況。
這裏還有一個不得不說的問題,就是關於XPath的。
對於這樣一個xml文檔,要查找第一個節點下的學生的Name時(ID=01),其XPath應該是"/ns:Root/ns:Students/ns:Student[1]/ns:Name"。xml對於重複的節點名稱,是按照順序1,2,3...的方式遍歷的,也就是說如果要找第N個Student節點的下的節點之,那麼應使用Student[N]的標識方式。
<?xml version="1.0" encoding="UTF-8" ?>
<Root xmlns="urn:ClassNameSpace">
<Class>
<ClassID>1234</ClassID>
</Class>
<Students>
<Student>
<ID>01</ID><Name>Name01</Name>
</Student>
<Student>
<ID>02</ID><Name>Name02</Name>
</Student>
</Students>
</Root>
當然,這裏也可以獲取節點屬性的值,查找滿足特定值的節點等等,這些和上面獲取節點值的過程是類似的。這裏推薦一篇介紹xpath的文章XPath 教程,大家不妨看看,關於xpath的常見問題都可以得到解決。