C#流模型详细分析

开发 后端
这里介绍C#流模型每次迭代XML文档中的一个节点,适合于处理较大的文档,所耗内存空间小。流模型中有两种变体——“推”模型和“拉”模型。

在向大家详细介绍C#流模型之前,首先让大家了解下在C#中如何使用XML,然后全面介绍C#流模型。

对于XML,想必各位都比较了解,我也就不用费笔墨来描述它是什么了,我想在未来的Web开发中XML一定会大放异彩,XML是可扩展标记语言,使用它企业可以制定一套自己的数据格式,数据按照这种格式在网络中传输然后再通过XSLT将数据转换成用户期望的样子表示出来,这样便轻易的解决了数据格式不兼容的问题。用于Internet的数据传输,我想,这是XML对于我们这些程序员最诱人的地方!

我们今天的主题不是论述XML的好处,而是讨论在C#中如何使用XML。下面我们来了解一下使用程序访问XML的一些基础理论知识。

访问的两种模型:

在程序中访问进而操作XML文件一般有两种模型,分别是使用DOM(文档对象模型)和流模型,使用DOM的好处在于它允许编辑和更新XML文档,可以随机访问文档中的数据,可以使用XPath查询,但是,DOM的缺点在于它需要一次性的加载整个文档到内存中,对于大型的文档,这会造成资源问题。流模型很好的解决了这个问题,因为它对XML文件的访问采用的是流的概念,也就是说,任何时候在内存中只有当前节点,但它也有它的不足,它是只读的,仅向前的,不能在文档中执行向后导航操作。虽然是各有千秋,但我们也可以在程序中两者并用实现优劣互补嘛,呵呵,这是题外话了!我们今天主要讨论XML的读取,那我们就详细讨论一下C#流模型吧!

C#流模型中的变体:

C#流模型每次迭代XML文档中的一个节点,适合于处理较大的文档,所耗内存空间小。流模型中有两种变体——“推”模型和“拉”模型。

推模型也就是常说的SAX,SAX是一种靠事件驱动的模型,也就是说:它每发现一个节点就用推模型引发一个事件,而我们必须编写这些事件的处理程序,这样的做法非常的不灵活,也很麻烦。

.NET中使用的是基于“拉”模型的实现方案,“拉”模型在遍历文档时会把感兴趣的文档部分从读取器中拉出,不需要引发事件,允许我们以编程的方式访问文档,这大大的提高了灵活性,在性能上“拉”模型可以选择性的处理节点,而SAX每发现一个节点都会通知客户机,从而,使用“拉”模型可以提高Application的整体效率。在.NET中“拉”模型是作为XmlReader类实现的,下面看一下该类的继承结构:

我们今天来讲一下该体系结构中的XmlTextReader类,该类提供对Xml文件进行读取的功能,它可以验证文档是否格式良好,如果不是格式良好的 Xml文档,该类在读取过程中将会抛出XmlException异常,可使用该类提供的一些方法对文档节点进行读取,筛选等操作以及得到节点的名称和值,请牢记:XmlTextReader是基于C#流模型的实现,打个不恰当的比喻,XML文件就好象水源,闸一开水就流出,流过了就流过了不会也不可以往回流。内存中任何时候只有当前节点,你可以使用 XmlTextReader类的Read()方法读取下一个节点。好了,说了这么多来看一个例子,编程要注重实际对吧。看代码前先看下运行效果吧!

Example1按纽遍历文档读取数据,Example2,Example3按纽得到节点类型,Example4过滤文档只获得数据内容,Example5得到属性节点,Example6按纽得到命名空间,Example7显示整个XML文档,为此,我专门写一个类来封装以上功能,该类代码如下:

  1. namespace XMLReading  
  2. {  
  3. using System;  
  4. using System.Xml;  
  5. using System.Windows.Forms;  
  6. using System.ComponentModel;  
  7.  
  8. /// <summary>  
  9. /// Xml文件读取器  
  10. /// </summary>  
  11.  
  12. public class XmlReader : IDisposable  
  13. {  
  14. private string _xmlPath;  
  15. private const string _errMsg = "Error Occurred While Reading ";  
  16. private ListBox _listBox;  
  17. private XmlTextReader xmlTxtRd;  
  18.  
  19. #region XmlReader 的构造器  
  20.  
  21. public XmlReader()  
  22. {  
  23. this._xmlPath = string.Empty;  
  24. this._listBox = null;  
  25. this.xmlTxtRd = null;  
  26. }  
  27.  
  28. /// <summary>  
  29. /// 构造器  
  30. /// </summary>  
  31. /// <param name="_xmlPath">xml文件绝对路径</param>  
  32. /// <param name="_listBox">列表框用于显示xml</param>  
  33.  
  34. public XmlReader(string _xmlPath, ListBox _listBox)  
  35. {  
  36. this._xmlPath = _xmlPath;  
  37. this._listBox = _listBox;  
  38. this.xmlTxtRd = null;  
  39. }  
  40.  
  41. #endregion  
  42. #region XmlReader 的资源释放方法  
  43.  
  44. /// <summary>  
  45. /// 清理该对象所有正在使用的资源  
  46.  
  47. /// </summary>  
  48.  
  49. public void Dispose()  
  50. {  
  51. this.Dispose(true);  
  52. GC.SuppressFinalize(this);  
  53. }  
  54.  
  55. /// <summary>  
  56. /// 释放该对象的实例变量  
  57. /// </summary>  
  58. /// <param name="disposing"></param>  
  59.  
  60. protected virtual void Dispose(bool disposing)  
  61. {  
  62. if (!disposing)  
  63. return;  
  64. if (this.xmlTxtRd != null)  
  65. {  
  66. this.xmlTxtRd.Close();  
  67. this.xmlTxtRd = null;  
  68. }  
  69.  
  70. if (this._xmlPath != null)  
  71. {  
  72. this._xmlPath = null;  
  73. }  
  74. }  
  75.  
  76. #endregion  
  77. #region XmlReader 的属性  
  78.  
  79. /// <summary>  
  80. /// 获取或设置列表框用于显示xml  
  81. /// </summary>  
  82.  
  83. public ListBox listBox  
  84. {  
  85. get  
  86. {  
  87. return this._listBox;  
  88. }  
  89. set  
  90. {  
  91. this._listBox = value;  
  92. }  
  93. }  
  94.  
  95. /// <summary>  
  96. /// 获取或设置xml文件的绝对路径  
  97. /// </summary>  
  98.  
  99. public string xmlPath  
  100. {  
  101. get  
  102. {  
  103. return this._xmlPath;  
  104. }  
  105. set  
  106. {  
  107. this._xmlPath = value;  
  108. }  
  109. }  
  110.  
  111. #endregion  
  112.  
  113. /// <summary>  
  114. /// 遍历Xml文件  
  115. /// </summary>  
  116.  
  117. public void EachXml()  
  118. {  
  119. this._listBox.Items.Clear();  
  120. this.xmlTxtRd = new XmlTextReader(this._xmlPath);  
  121.  
  122. try  
  123. {  
  124. while(xmlTxtRd.Read())  
  125. {  
  126. this._listBox.Items.Add(this.xmlTxtRd.Value);  
  127. }  
  128. }  
  129. catch(XmlException exp)  
  130. {  
  131. throw new XmlException(_errMsg + this._xmlPath + exp.ToString());  
  132. }  
  133. finally  
  134. {  
  135. if (this.xmlTxtRd != null)  
  136. this.xmlTxtRd.Close();  
  137. }  
  138. }  
  139.  
  140. /// <summary>  
  141. /// 读取Xml文件的节点类型  
  142. /// </summary>  
  143.  
  144. public void ReadXmlByNodeType()  
  145. {  
  146. this._listBox.Items.Clear();  
  147. this.xmlTxtRd = new XmlTextReader(this._xmlPath);  
  148.  
  149. try  
  150. {  
  151. while(xmlTxtRd.Read())  
  152. {  
  153. this._listBox.Items.Add(this.xmlTxtRd.NodeType.ToString());  
  154. }  
  155. }  
  156. catch(XmlException exp)  
  157. {  
  158. throw new XmlException(_errMsg + this._xmlPath + exp.ToString());  
  159. }  
  160. finally  
  161. {  
  162. if (this.xmlTxtRd != null)  
  163. this.xmlTxtRd.Close();  
  164. }  

【编辑推荐】

  1. C#方法重写全面介绍
  2. C#流程控制语句简单描述
  3. Java和C#字符串类型概述
  4. C#访问修饰符详细剖析
  5. 选择C#构造函数描述
责任编辑:佚名 来源: 博客园
相关推荐

2009-09-07 13:19:44

C#线程同步

2009-09-03 17:57:06

C#声明事件

2009-09-07 14:18:01

C#内存管理

2009-09-14 13:50:35

LINQ编程模型

2009-08-10 17:34:42

C#数据库连接池

2009-09-25 14:23:39

2009-09-28 10:39:01

Hibernate基础

2009-09-14 16:21:34

LINQ To XML

2009-09-09 09:48:43

Linq延迟加载

2009-06-18 14:00:51

2009-10-10 13:52:57

VB Update方法

2009-09-08 15:56:50

Linq使用Group

2009-11-20 13:11:44

Oracle XML数

2010-01-06 13:50:37

.NET Framew

2009-03-24 08:30:54

AndroidGoogle移动os

2009-12-16 14:09:14

Visual Stud

2009-12-07 15:37:00

WCF控件

2010-04-26 18:17:19

Oracle存储过程

2011-09-24 12:34:03

2009-09-09 13:53:21

Linq表值函数
点赞
收藏

51CTO技术栈公众号