自己编写XPath练习器

类别:.NET开发 点击:0 评论:0 推荐:

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

本文地址:http://com.8s8s.com/it/it44805.htm