2025年3月31日 星期一

[研究]ASP.NET 使用 NPOI 匯出寫入 Excel (.xlsx) 檔案

[研究]ASP.NET 使用 NPOI 匯出寫入 Excel (.xlsx) 檔案

[研究]ASP.NET 把 ClosedXML 匯出寫入改寫為使用 NPOI 匯出寫入 Excel (.xlsx) 檔案

2025-03-31

環境:Visual Studio 2022 + ASP.NET + WebForm + Web Application + C# + SQL Server 2019 + SQL Server Management Studio (SSMS) 19

********************************************************************************

Default.aspx

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Default.aspx.cs"
    Inherits="WebApplication1.Default" %>

<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
    <title></title>
</head>
<body>
    <form id="form1" runat="server">
        <asp:Button ID="Button_Export_Xlsx_by_ClosedXML" runat="server"
            OnClick="Button_Export_Xlsx_by_ClosedXML_Click" Text="匯出(ClosedXML)"
            CssClass="btn btn-info btn-xs" /><br />
        <asp:Button ID="Button_Export_Xlsx_by_NPOI" runat="server"
            OnClick="Button_Export_Xlsx_by_NPOI_Click" Text="匯出(NPOI)"
            CssClass="btn btn-info btn-xs" /><br />
        <asp:Label ID="Label_MSG1" runat="server"></asp:Label>
        <br />
    </form>
</body>
</html>


Default.aspx.cs

using ClosedXML.Excel;
using System;
using System.Configuration;
using System.Data.SqlClient;
using System.IO;
using System.Linq;
using System.Text;
using System.Web;
using System.Web.Configuration;
using System.Web.UI;
using System.Web.UI.WebControls;

using NPOI.SS.UserModel;
using NPOI.XSSF.UserModel;

namespace WebApplication1
{
    public partial class Default : System.Web.UI.Page
    {
        protected void Page_Load(object sender, EventArgs e)
        {

        }

        // 匯出查詢結果(.xlsx)
        #region == protected void Button_Export_Xlsx_by_ClosedXML_Click(object sender, EventArgs e) ==
        protected void Button_Export_Xlsx_by_ClosedXML_Click(object sender, EventArgs e)
        {
            //string mainFileName = "匯出檔案名稱";
            string mainFileName = "匯出";
            string fd = (string)ConfigurationManager.AppSettings["TempFolder"] + Path.DirectorySeparatorChar.ToString();
            string dateTimeString = DateTime.Now.ToString("yyyy-MM-dd-HH-mm-ss-fff");
            // 若檔案名稱有中文字或特殊符號,Response.AddHeader() 輸出檔案名稱要編碼
            string outFileName = HttpUtility.UrlEncode(
                mainFileName, System.Text.Encoding.UTF8) + "-" +
                dateTimeString + ".xlsx";

            // XLWorkbook.SaveAs() 輸出檔案名稱不要用 HttpUtility.UrlEncode 編碼
            // Response.TransmitFile() 輸出檔案名稱不要用 HttpUtility.UrlEncode 編碼
            string dirFileName = fd + mainFileName + "-" +
                dateTimeString + ".xlsx";
            try
            {
                Response.Clear();
                Response.Buffer = true;
                Response.Charset = "";

                #region === Export ===

                // [EXCEL] Excel打開是亂碼?快速找回資料的最好方法!
                // http://blog.e-happy.com.tw/excel-excel%E6%89%93%E9%96%8B%E6%98%AF%E4%BA%82%E7%A2%BC%EF%BC%9F%E5%BF%AB%E9%80%9F%E6%89%BE%E5%9B%9E%E8%B3%87%E6%96%99%E7%9A%84%E6%9C%80%E5%A5%BD%E6%96%B9%E6%B3%95%EF%BC%81/

                // Excel開啟CSV時的中文編碼問題補遺
                //https://blog.darkthread.net/blog/csv-encoding-again/

                //Response.ContentType = "application/vnd.oasis.opendocument.spreadsheet";    // ODF/.ods
                //Response.AddHeader("content-disposition", "attachment;filename=" + outFileName);
                using (MemoryStream myMemoryStream = new MemoryStream())
                {


                    string dataText = "";

                    var wb = new XLWorkbook();
                    var ws = wb.Worksheets.Add("工作表1");

                    // 定義欄位名稱列
                    string[] columnNameArray = { "欄位1", "欄位2", "欄位3" };
                    // Column number must be between 1 and 16384
                    for (int i = 0; i < columnNameArray.Count(); i++)
                    {
                        ws.Cell(1, i + 1).Value = columnNameArray[i];
                    }

                    // 讀取資料,資料寫入「工作表1」
                    //                    string queryString = @"
                    //--DECLARE  @FieldText nvarchar(50)
                    //--SET @FieldText=N'abc'

                    //SELECT * FROM [TestDB].[dbo].[Table1]
                    //";

                    //string queryString = SqlDataSource1.SelectCommand;
                    string queryString = @"SELECT * FROM [MyDB].[dbo].[MyTable]";

                    //queryString = SqlDataSource1.SelectCommand;
                    using (SqlConnection connection = new SqlConnection(
                        WebConfigurationManager.ConnectionStrings["MyDBConnectionString"].ConnectionString))
                    {
                        SqlCommand command = new SqlCommand(queryString, connection);
                        connection.Open();
                        using (SqlDataReader reader = command.ExecuteReader())
                        {
                            int rowIndex = 1;   // 1 是標題列
                            while (reader.Read())
                            {
                                rowIndex++;
                                for (int i = 0; i < reader.FieldCount; i++)
                                {
                                    //reader[i]的 i 要從 0 開始
                                    dataText = reader[i].ToString();

                                    // 規則運算式語言 - 快速參考 | Microsoft Docs
                                    // https://docs.microsoft.com/zh-tw/dotnet/standard/base-types/regular-expression-language-quick-reference

                                    // 換掉特殊字元 (自己評估)
                                    dataText = dataText.Replace("\r", "").Replace("\n", "").Replace("\t", "").Replace(",", ",").Replace("'", "′").Replace(@"""", "”");
                                    dataText = dataText.Replace("\a", "").Replace("\b", "").Replace("\t", "").Replace("\v", "").Replace("\f", "").Replace("\\", "");

                                    // Column number must be between 1 and 16384
                                    ws.Cell(rowIndex, i + 1).Value = dataText;
                                }
                            }
                        }

                        Response.Clear();
                        Response.Buffer = true;
                        Response.ContentEncoding = Encoding.GetEncoding(950);   //950就是所謂的BIG5
                                                                                //Response.AddHeader("Content-Disposition", "attachment;filename=\"" +
                                                                                //    HttpUtility.UrlEncode(fileName, System.Text.Encoding.UTF8) + "-" +
                                                                                //    DateTime.Now.ToString("yyyyMMddHHmmss") + ".xlsx");
                        Response.AddHeader("Content-Disposition", "attachment;filename=\"" + outFileName);
                        //'Excel 2003 : "application/vnd.ms-excel"
                        //'Excel 2007 : "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"

                        //Extension       MIME Type
                        //.xlsx               application/vnd.openxmlformats-officedocument.spreadsheetml.sheet
                        //.xltx                application/vnd.openxmlformats-officedocument.spreadsheetml.template
                        //.potx               application/vnd.openxmlformats-officedocument.presentationml.template
                        //.ppsx              application/vnd.openxmlformats-officedocument.presentationml.slideshow
                        //.pptx               application/vnd.openxmlformats-officedocument.presentationml.presentation
                        //.sldx               application/vnd.openxmlformats-officedocument.presentationml.slide
                        //.docx              application/vnd.openxmlformats-officedocument.wordprocessingml.document
                        //.dotx               application/vnd.openxmlformats-officedocument.wordprocessingml.template
                        //.xlam              application/vnd.ms-excel.addin.macroEnabled.12
                        //.xlsb               application/vnd.ms-excel.sheet.binary.macroEnabled.12

                        //Response.ContentType = "application/ms-excel";
                        Response.ContentType = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet";

                        //Response.TransmitFile(@"C:\Program Files (x86)\IIS Express\Export.xlsx");
                        //dirFileName = dirFileName.Replace('\\', '/');   //轉換 Windows, Linux 不同路徑斜線
                        //dirFileName = dirFileName.Replace("/", "\\");  // 轉換斜線 / 成 \  ,因 \ 是特殊符號,要以 \\ 表示

                        // 方法1: 直接輸出
                        wb.SaveAs(myMemoryStream);
                        Response.BinaryWrite(myMemoryStream.ToArray());
                        // myMemoryStream.WriteTo(Response.OutputStream); //works too
                        Response.Flush();
                        Response.Close();

                        // 方法2: 若需要寫入檔案再輸出
                        wb.SaveAs(dirFileName);                 // 存檔,Debug 或其他用途
                                                                ////Response.WriteFile(savePath);
                                                                //Response.TransmitFile(dirFileName);   // 把存檔輸出

                        // 輸出結束
                        Response.End();

                        Label_MSG1.ForeColor = System.Drawing.Color.Green;
                        Label_MSG1.Text = "匯出成功。";
                        // 上面 Response 輸出檔案,所以下面 JavaScript alert 是不會跳出的
                        //Page.ClientScript.RegisterStartupScript(Page.GetType(), "message", "<script language='javascript' defer>alert('匯出成功。');</script>");
                    }
                }
                #endregion Export ODS
            }
            catch (Exception)
            {
                throw;
            }
        }
        #endregion

        // 匯出查詢結果(.xlsx)
        #region == protected void Button_Export_Xlsx_by_NPOI_Click(object sender, EventArgs e) ==
        protected void Button_Export_Xlsx_by_NPOI_Click(object sender, EventArgs e)
        {
            string mainFileName = "匯出";
            string dateTimeString = DateTime.Now.ToString("yyyy-MM-dd-HH-mm-ss-fff");
            string outFileName = HttpUtility.UrlEncode(mainFileName, System.Text.Encoding.UTF8) + "-" + dateTimeString + ".xlsx";

            try
            {
                using (MemoryStream ms = new MemoryStream())
                {
                    IWorkbook workbook = new XSSFWorkbook();  // 使用 XSSFWorkbook (XLSX 格式)
                    ISheet sheet = workbook.CreateSheet("工作表1");

                    // 設定標題列
                    string[] columnNames = { "欄位1", "欄位2", "欄位3" };
                    IRow headerRow = sheet.CreateRow(0);
                    for (int i = 0; i < columnNames.Length; i++)
                    {
                        headerRow.CreateCell(i).SetCellValue(columnNames[i]);
                    }

                    // 讀取資料
                    string queryString = @"SELECT * FROM [MyDB].[dbo].[MyTable]";
                    using (SqlConnection connection = new SqlConnection(
                        WebConfigurationManager.ConnectionStrings["MyDBConnectionString"].ConnectionString))
                    {
                        SqlCommand command = new SqlCommand(queryString, connection);
                        connection.Open();
                        using (SqlDataReader reader = command.ExecuteReader())
                        {
                            int rowIndex = 1;
                            while (reader.Read())
                            {
                                IRow row = sheet.CreateRow(rowIndex++);
                                for (int i = 0; i < reader.FieldCount; i++)
                                {
                                    string dataText = reader[i].ToString()
                                        .Replace("\r", "").Replace("\n", "").Replace("\t", "")
                                        .Replace(",", ",").Replace("'", "′").Replace("\"", "”")
                                        .Replace("\\", "").Replace("\a", "").Replace("\b", "")
                                        .Replace("\v", "").Replace("\f", "");

                                    row.CreateCell(i).SetCellValue(dataText);
                                }
                            }
                        }
                    }

                    // **避免 MemoryStream 被 NPOI 關閉**
                    workbook.Write(ms);
                    ms.Flush();

                    // 設定 Response 參數
                    Response.Clear();
                    Response.Buffer = true;
                    Response.ContentEncoding = Encoding.UTF8;
                    Response.AddHeader("Content-Disposition", "attachment;filename=\"" + outFileName + "\"");
                    Response.ContentType = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet";

                    // 輸出檔案
                    Response.BinaryWrite(ms.ToArray());
                    Response.Flush();
                    Response.End();
                }
            }
            catch (Exception ex)
            {
                Label_MSG1.ForeColor = System.Drawing.Color.Red;
                Label_MSG1.Text = "匯出失敗:" + ex.Message;
            }
        }


        #endregion
    }
}

敝人實際20萬筆測試速度,有時ClosedXML快,有時 NPOI快。

(完)

相關

沒有留言:

張貼留言