Asp.net中把DataTable或DataGrid導出爲Excel

Asp.net中把DataTable或DataGrid導出爲Excel

當前編碼的一個項目中有把查詢結果(顯示在DataGrid)導出爲excel的需求,嘗試了幾種方法,作爲技巧拿來和大家分享。

內容:

服務器端實現DataGrid導出爲excel

客戶端實現DataGrid導出爲excel

服務器端實現DataTable導出爲excel(終極解決方案)

--------------------------------------------------------------------------------

服務器端實現DataGrid導出爲excel

這是網上出現的最多的做法:

1/**//// <summary>
2    /// 把DataGrid內容導出偉excel並返回客戶端
3    /// </summary>
4    /// <param name="dgData">待導出的DataGrid</param>
5    /// 創 建 人:calvin
6    /// 創建日期:2005年10月08日
7    /// 修 改 人:
8    /// 修改日期:
9    public static void DataGrid2Excel(System.Web.UI.WebControls.DataGrid dgData)
10    {
11      // 當前對話
12      System.Web.HttpContext curContext = System.Web.HttpContext.Current;
13      // IO用於導出並返回excel文件
14      System.IO.StringWriter strWriter = null;
15      System.Web.UI.HtmlTextWriter htmlWriter = null;
16
17      if (dgData != null)
18      {
19        // 設置編碼和附件格式
20        curContext.Response.ContentType = "application/vnd.ms-excel";
21        curContext.Response.ContentEncoding =System.Text.Encoding.UTF8;
22        curContext.Response.Charset = "";
23        
24        // 導出excel文件
25        strWriter = new System.IO.StringWriter();
26        htmlWriter = new System.Web.UI.HtmlTextWriter(strWriter);
27
28        // 返回客戶端  
29        dgData.RenderControl(htmlWriter);  
30        curContext.Response.Write(strWriter.ToString());
31        curContext.Response.End();
32      }
33    }

在需要導出的地方調用上面的方法就可以。不過這樣的實現有兩個問題:第一,datagrid中不能包含模板列;第二,只能夠導出當前顯示在datagrid的數據,無法在分頁的情況下導出全部的查詢結果。

如果大家分析一下Control.RenderControl的方法,就會發現RenderControl只是把控件的innerHTML導出來,既然如此,完全可以把導出操作放在客戶端來處理。

--------------------------------------------------------------------------------

客戶端導出excel

1 /*
2 * 將DataGrid導出爲Excel文件
3 *
4 * @param strTitle  文件標題
5 * @param dgData    待導出的DataGrid
6 * @param iStartCol  起始列序號
7 * @param iEndCol  結束列序號
8 *
9 * 創建人:  calvin
10 * 創建日期: 2005-10-08
11 * 修改人:  
12 * 修改日期:
13 **/
14 function DataGrid2Excel(strTitle, dgData, iStartCol, iEndCol)
15 {
16   // 定義Excel Applicaiton Object
17   var appExcel = null;
18   // 當前激活的工作簿
19   var currentWork = null;
20   var currentSheet = null;
21   
22   try
23   {
24     // 初始化application
25     appExcel = new ActiveXObject("Excel.Application");
26     appExcel.Visible = true;
27   }
28   catch(e)
29   {
30     window.alert("Please Install Excel First");
31     
32     return;
33   }
34   
35   // 獲取當前激活的工作部
36   currentWork = appExcel.Workbooks.Add();
37   currentSheet = currentWork.ActiveSheet;
38 
39   // 填充excel內容
40   // 設置標題
41   currentSheet.Cells(1,1).Value = strTitle;
42   currentSheet.Cells(1,1).Value = dgData.innerText;
43   window.alert(dgData.innerHTML);
44 
45   // 填充內容
46   for (var iRow = 0; iRow < dgData.rows.length - 1; iRow++)
47   {
48     // 顯示指定列的內容
49     for (var iCol = iStartCol; iCol <= iEndCol; iCol++)
50     {
51       currentSheet.Cells(iRow + 2, iCol + 1).Value = 
52         dgData.rows[iRow].cells[iCol].innerText;
53     }
54   }
55 }
下面是調用的例子
1 /**
2 * 導出dgData中0-3列的數據到excel文件中
3 **/
4 function ToExcel()
5 {
6   DataGrid2Excel("使用javascript導出excel的例子", document.getElementsById("dgData"), 0, 3);
7 }
這種方法的缺點是:

(1)了能夠在客戶端調用Excel.Application,需要把IE的安全級別設爲“低”。

(2)與方法一相同,還是隻能導出當前顯示在datagrid裏面的數據,無法導出分頁的數據。

--------------------------------------------------------------------------------

終極解決方案:將DataTable導出爲excel

好,讓我們快點結束這篇無聊的post。一般來說,頁面上的datagrid是以查詢得到的一個DataTable爲數據源的。那麼爲了把全部數據導入excel中,我們只要把DataTable數據源輸出爲excel就可以了。

1/**//// <summary>
2    /// 把DataTable內容導出偉excel並返回客戶端
3    /// </summary>
4    /// <param name="dgData">待導出的DataTable</param>
5    /// 創 建 人:陳文凱
6    /// 創建日期:2005年10月08日
7    /// 修 改 人:
8    /// 修改日期:
9    public static void DataTable2Excel(System.Data.DataTable dtData)
10    {
11      System.Web.UI.WebControls.DataGrid dgExport = null;
12      // 當前對話
13      System.Web.HttpContext curContext = System.Web.HttpContext.Current;
14      // IO用於導出並返回excel文件
15      System.IO.StringWriter strWriter = null;
16      System.Web.UI.HtmlTextWriter htmlWriter = null;
17
18      if (dtData != null)
19      {
20        // 設置編碼和附件格式
21        curContext.Response.ContentType = "application/vnd.ms-excel";
22        curContext.Response.ContentEncoding =System.Text.Encoding.UTF8;
23        curContext.Response.Charset = "";
24        
25        // 導出excel文件
26        strWriter = new System.IO.StringWriter();
27        htmlWriter = new System.Web.UI.HtmlTextWriter(strWriter);
28
29        // 爲了解決dgData中可能進行了分頁的情況,需要重新定義一個無分頁的DataGrid
30        dgExport = new System.Web.UI.WebControls.DataGrid();
31        dgExport.DataSource = dtData.DefaultView;
32        dgExport.AllowPaging = false;
33        dgExport.DataBind();
34
35        // 返回客戶端
36        dgExport.RenderControl(htmlWriter);  
37        curContext.Response.Write(strWriter.ToString());
38        curContext.Response.End();
39      }
40    }


  需要注意的是,導出excel之前要把datatable的列名更改爲客戶要求的文字,就ok了。因爲是從DataTable導出的,所以這種方法解決了分頁數據的問題,堪稱終極解決方案

 

一、DataTable To Excel
  先定義一下:
  private Excel.Application excelApp= new Excel.ApplicationClass();//Application與ApplicationClass的區別我沒有明白。
  private Excel.Workbook excelBook = excelApp.Workbooks.Add(Type.Missing);
  private Excel.Worksheet excelSheet = (Excel.Worksheet)excelBook.ActiveSheet;
  excelApp.Visible = true;
  1、讓我們看看在網上流傳最多的解決方案:
  public void DataTableToExcel(DataTable dt)
  {
  for (int i = 0; i <dt.Rows.Count; i++)
  {
  for (int j=0;j<dt.Columns.Count;j++)
  {
  excelSheet.Cells[i+1,j+1] = dt.Rows[i][j].ToString();
  }
  }
  }
  不可否認這段程序會不出錯誤的運行,但是效率就比較低了,會隨着數據行和列的增加而增加,主要的時間都耗在了Excel單元格的讀取上。
  讓我們改進一下,利用DataRow.ItemArray:
  public void DataTableToExcel(DataTable dt)
  {
  int colCount = dt.Columns.Count;
  for (int i = 0; i <dt.Rows.Count; i++)
  {
  excelSheet.get_Range(excelSheet.Cells[i+1,1], excelSheet.Cells[i+1, colCount])。Value2 = dt.Rows[i].ItemArray;
  }
  這就有了很大進步,效率與數據列的多少已經無關了,對於這個改進我在網上還沒有看到,爲此我還曾沾沾自喜,但是這個方法的效率依然不高,尤其是靈活性不夠。
  以上效率的損失主要是每導出一條記錄都要與Excel交互一次,如果我們把數據準備好一次性寫入Excel是否效率更快?看如下程序:
  public void DataTableToExcel(DataTable dt)
  {
  int rowCount = dt.Rows.Count;
  int colCount = dt.Columns.Count;
  object[,] dataArray = new object[rowCount,colCount];
  for (int i = 0; i <rowCount; i++)
  {
  for (int j=0;j<colCount;j++)
  {
  dataArray[i, j] = dt.Rows[i][j];
  }
  }
  excelSheet.get_Range("A1", excelSheet.Cells[rowCount, colCount])。Value2 = dataArray;
  }
  由於二維數組是在內存中實現,所以速度很快,由於是一次性寫入到Excel中,使我們幾乎感覺不到延遲,在Excel打開後數據馬上就出來了,感覺很爽。
  這一篇先寫這些,下一篇中主要討論一下字段標題的設置技巧,行列的篩選與排序等的技巧。
  另,在上述學習過程中,我曾想能否利用dt.Rows[i].ItemArray來組建二維數組,也就是多個相同結構的一維數組簡單快速的合併成一個二維數組,當然不是利用for循環?

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章