这段时间在看XML,结果Asp.net中的XPath功能,写了一个小页面。一来可以学习Asp.Net中的操作XML文档的知识,二来更可以把这个页面当作以后来练习编写XPath表达式的工具。
在编写页面过程中,我们需要用到XmlDocument XmlNode XPathNavigator XPathExpression XPathNodeIterator等类。
XmlDocument类,在这里,我们用它来装载一个Xml文档,装载后,我们会使用Cache技术,将文档保存在内存中,避免重复装载相同的文档。
XPathNavigator类,在这个页面,它是主角了。XPathNavigator类是一个支持XPath的具有强大功能的类,这里,我们只用到它的几个主要属性和方法。首先,这个类的实例不是通过构造函数得到,而是通过XmlNode或XmlDocument的CreateNavigator方法得到。然后,你可以将一个XPath字符串交给XPathNavigator.Compile方法来得到一个XPathExpression类。接着,调用XPathNavigator.Select或XPathNavigator.Evaluate方法来完成XPath的运算过程,它们均接受一个XPathExpression参数。通常,为了让我们的程序具有较强的通用性或正确性,在调用XPathNavigator.Select或XPathNavigator.Evaluate方法之前,会先判断XPathExprssion实例在运算时的返回类型(它是一个XPathResultType枚举值),然后,根据返回类型的不同,进行相应的操作。它通过的类型有:
Any 任何一种 XPath 节点类型。
Boolean 布尔值 true 或 false。
Error 该表达式的计算结果不是正确的 XPath 类型。
Navigator 一个树片段。
NodeSet 一个节点集合。
Number 一个数值。
String 一个字符串值。
本程序中,我们只用到其中的一部分(稍后,在源程序中会看到)。
如果返回类型为NodeSet,我们则要用到XPathNodeIterator类,它是XPath运算结果集的迭代,通过MoveNext只读方法前移。
同时,对于返回多个结点的NodeSet的处理过程,我们还想得到具体的节点类型和它的内容。这时,我们通过((IHasXmlNode)XPathNodeIterator.Current).GetNode()方法得到迭代指针所指向的当前的XmlNode,通过XmlNode.NodeType和XmlNode.OuterXml得到较为详细的信息。
页面表单主要用到HtmlInputFile TextBox Button Label Span等控件或标记来完成文档上传和显示信息等工作。
详细的源代码如下:
<%@Page Debug="true"%>
<%@Import Namespace="System"%>
<%@Import Namespace="System.Xml"%>
<%@Import Namespace="System.Xml.XPath"%>
<%@Import Namespace="System.IO"%>
<Script Language="C#" runat="Server">
protected XmlDocument myXmlDoc;
protected XmlNode singleNode;
protected XmlNodeList moreNodes;
protected String strFileName;
protected void DoXPath(Object sender,EventArgs e)
{
if ((fileXmlDoc.PostedFile.FileName=="")&&(Cache["strFileName"]==null))
{
lblMessage.Text = "你没有选择XML文档";
return;
}
if (tbxExpression.Text==String.Empty)
{
lblMessage.Text = "XPath表达式不能为空";
return;
}
try
{
LoadXmlDoc();
XPathNavigator nav = myXmlDoc.CreateNavigator();
XPathExpression expr = nav.Compile(tbxExpression.Text.ToString().Trim());
lblResult.InnerHtml="<hr align='left' width='400'>";
switch(expr.ReturnType)
{
case XPathResultType.NodeSet:
//do返回节点集的情况
{
XPathNodeIterator iterator = nav.Select(expr);
if (iterator.Count==0)
{
lblMessage.Text="当前没有可匹配的节点或属性";
lblResult.InnerHtml=String.Empty;
}
else
{
lblMessage.Text="共匹配" + iterator.Count.ToString() + "个节点";
}
while(iterator.MoveNext())
{
XPathNavigator currentNode = iterator.Current;
String strNodeName = (currentNode.Name==String.Empty)?"No Name":currentNode.Name;
XmlNode xmlNode = (XmlNode)((IHasXmlNode)currentNode).GetNode();
String strNodeValue = (Server.HtmlEncode(xmlNode.OuterXml)==String.Empty)?"No Conent":Server.HtmlEncode(xmlNode.OuterXml);
String strNodeType = String.Empty;
switch(currentNode.NodeType)
{
case XPathNodeType.Element:
strNodeType = "当前匹配元素节点";
break;
case XPathNodeType.Attribute:
strNodeType = "当前匹配属性节点";
break;
case XPathNodeType.Comment:
strNodeType = "当前匹配注释节点";
break;
case XPathNodeType.Text:
strNodeType = "当前匹配文本节点";
break;
case XPathNodeType.Root:
strNodeType = "当前匹配根节点";
break;
}
lblResult.InnerHtml += strNodeType + "<br/>" + "<b>" + strNodeName + "</b> " + strNodeValue + " " + "<hr align='left' width='400'/>";
}
break;
}
case XPathResultType.Boolean:
//do返回布尔值
{
lblMessage.Text = "当前匹配结果为布尔值";
lblResult.InnerHtml += nav.Evaluate(expr).ToString()+"<hr align='left' width='400'/>";
break;
}
case XPathResultType.Number:
//do返回数值
{
lblMessage.Text = "当前匹配结果为数值";
lblResult.InnerHtml += nav.Evaluate(expr).ToString()+"<hr align='left' width='400'/>";
break;
}
case XPathResultType.String:
//do返回字符串
{
lblMessage.Text = "当前匹配结果为字符串";
lblResult.InnerHtml += nav.Evaluate(expr).ToString() + "<hr align='left' width='400'/>";
break;
}
default:
{
lblMessage.Text="无法确定表达的返回结果类型";
lblResult.InnerHtml="";
break;
}
}
}
catch(Exception excep)
{
lblMessage.Text = excep.Message;
lblResult.InnerHtml="";
}
}
protected void LoadXmlDoc()
{
strFileName = fileXmlDoc.PostedFile.FileName;
//如果文件上传框的内容有改变,则重新装载文档。
//如果没有缓存文档,也要重新装载文档。
if ( ((Cache["strFileName"]!=strFileName)&&(strFileName!=String.Empty)) ||
(Cache["strFileName"]==null) )
{
myXmlDoc = new XmlDocument();
myXmlDoc.Load(fileXmlDoc.PostedFile.InputStream);
Cache["strFileName"] = strFileName;
Cache["myXmlDoc"] = myXmlDoc;
return;
}
else
{
myXmlDoc = (XmlDocument)Cache["myXmlDoc"];
return;
}
}
</Script>
<html>
<form runat="server">
<H3>XPath Expression</H3>
<span style="font-size:8pt;background-color:#cccccc">提示:只要不关闭本页面,第一次提交的XML文档始终有效,直到你再次提交新的文档。</span>
<br/>
<span style="font-size:10pt">Xml文件</span>
<input type="file" width="100px" size="40" id="fileXmlDoc" runat="server"/>
<br/>
<span style="font-size:10pt">XPath表达式</span>
<asp:TextBox id="tbxExpression" Font-Size="10pt" BorderStyle="Inset"
Size="40" BorderColor="Black" BorderWidth="1px" runat="server"/>
<br/>
<asp:Button id="btnExecuteXPath" OnClick="DoXPath" Text="匹配表达式" Font-Size="10pt" runat="server"/>
<br/>
<asp:Label id="lblMessage" BackColor="#aaccee" Font-Size="10pt" Width="600" BorderStyle="Inset" BorderColor="Black"
BorderWidth="1px" runat="server"/>
<br/>
<table width="600px">
<tr width="100%">
<td>
<span name="lblResult" id="lblResult" style="font-size:10pt" runat="server"/>
</td>
<tr>
</table>
</form>
</html>
上面的功能还比较粗糙,希望大家能多提建议或给出完善的方法。同时,我没有找到按缩进格式来显示XML片断的方法,希望能有朋友提供相应的方法。谢谢!