这段时间在看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片断的方法,希望能有朋友提供相应的方法。谢谢! 
|