ASP.NET中应用Excel:(3)访问工作表数据

已经创建了Excel.Application对象,并打开了工作薄,现在可以遍历工作表和单元格来获取数据了。并不是所有的单元格都有数据,大部分都是空的,因此XmlDocument来保存取得的数据是个不错的主意,还可以将XmlDocuemnt作为ScriptMethod的结果返回到客户端浏览器,使用JavaScript来生成表格,表现结果。根据我的经验,即使在服务端创建表格,也只消耗Excel操作的数百分之一的时间(实际数据:Excel操作耗时22.3秒,生成表格0.06秒,差距巨大)。

可以通过以下方式遍历工作表:

[csharp] view plaincopy
  1. sheets = workbook.Sheets;  
  2.   
  3.   
  4.   
  5. for (int i = 1; i <= sheets.Count; i++) // 注意Excel的对象数组访问是以1为起始滴!  
  6.   
  7. {  
  8.   
  9.       Excel.Worksheet sheet = (Excel.Worksheet)sheets.get_Item(i);// 获取工作表对象  
  10.   
  11.       ......  
  12.   
  13.       releaseComObject(sheet);  // 释放COM对象  
  14.   
  15.       sheet = null;  
  16.   
  17. }  

Excel的表格范围从A-IV列共256列,65536行,而且很多单元格是空的,完全遍历会消耗很多时间,因此,我们应该限定一个范围:

[csharp] view plaincopy
  1. Excel.Range range = sheet.get_Range("A1""Z128"); // 范围为A1到Z128,注意Excel.Range对象会产生引用计数  
  2.   
  3. Excel.Range rowRange = range.Rows;  //  行对象,带"s",会产生引用计数  
  4.   
  5. Excel.Range colRange = range.Columns; // 列对象,也会产生引用计数  
  6.   
  7.   
  8.   
  9. object[,] data = (object[,])range.get_Value(Type.Missing); // 一次性获取范围内的所有数据,注意,获取的是值!  
  10.   
  11.   
  12.   
  13. for (int r = rowRange.Count; r >= 1; r--) // 使用倒序遍历行  
  14.   
  15. {  
  16.   
  17.     for ( int c = 1; c <= colRange.Count; c++ ) // 顺序访问列  
  18.   
  19.     {  
  20.   
  21.         if ( data[r, c] != null ) // 如果[r,c]位置的单元格没有数据,则值为null,这可以检测空单元格  
  22.   
  23.          {  
  24.   
  25.              Excel.Range cell = (Excel.Range)range.get_Item(r, c);  
  26.   
  27.              ......  
  28.   
  29.                
  30.   
  31.              releaseComObject(cell); // 释放引用  
  32.   
  33.                cell = null;  
  34.   
  35.         }  
  36.   
  37.     }  
  38.   
  39. }  

在Excel 2003中,通过cell.Value属性访问值域,在Excel 2007中,则为cell.Value2。

对于以倒序的方式遍历行的方式的解释: 假设我们的XML格式如下: <WORKSHEET NAME="SheetXXX">     <ROW>         <CELL>XXXXXXX</CELL>     </ROW> </WORKSHEET> 由于Excel工作表数据集中在头部,后面基本是空的,填充XML时先生成CELL元素,插入到ROW元素中,再将ROW插入到WORKSHEET元素中。当以倒序遍历行时,如果行为空,不包括单元格数据,则ROW的ChildNodes.Count为0,如果这时WORKSHEET的ChildNodes为空,表示不包含行数据,则不需要将ROW插入到WORKSHEET中。这种情况直到遇到第一个包含数据单元格的行后改变。这时,行会添加到WORKSHEET的ChildNodes中。通过这种方式,可以跳过尾部没有数据的行,又不会略过夹在有数据的行中间的空行。

插入行元素的示例代码如下:

[csharp] view plaincopy
  1. if (row_elem.ChildNodes.Count > 0 || sheet_elem.ChildNodes.Count > 0) // 有行数据或空行则插入  
  2.   
  3. {  
  4.   
  5.     sheet_elem.InsertBefore(row_elem, sheet_elem.FirstChild);  
  6.   
  7. }  
发布了5 篇原创文章 · 获赞 23 · 访问量 10万+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章