在EPPlus导出中检测数据表日期字段和强制日期格式

时间:2015-12-07 15:43:08

标签: c# asp.net epplus

我的页面上有一个按钮,点击后会将数据从gridview导出为ex​​cel。

protected void btExport_Click(object sender, EventArgs e)
{
    Response.Clear();
    Response.Charset = "";
    Response.ContentEncoding = System.Text.Encoding.UTF8;
    Response.Cache.SetCacheability(HttpCacheability.NoCache);
    Response.ContentType = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet";
    Response.AddHeader("content-disposition", "attachment;filename=Output.xlsx");

    DataTable dt= gvOutput.DataSource as DataTable;

    ExcelPackage pck = new ExcelPackage();
    using(pck)
    {
      ExcelWorksheet wsDt = pck.Workbook.Worksheets.Add("Sheet1");
      wsDt.Cells["A1"].LoadFromDataTable(dt, true, TableStyles.None);
      wsDt.Cells[wsDt.Dimension.Address].AutoFitColumns();

      Response.BinaryWrite(pck.GetAsByteArray());

    }

    Response.Flush();
    Response.End();
}

使用上面的代码可以正常工作,但日期字段的格式为整数。我理解特定单元格可以被强制使用某种格式,但我的问题是日期列会随着gridview的数据集是动态的而改变。

我需要一种方法来动态识别哪一列是日期,并强制该列的格式为日期时间。

我想它会涉及这样的IF声明,但我还没有弄清楚要放入什么或放在哪里!

if (dt.Columns[x].DataType == typeof(DateTime))
{ 
    //do something
}

非常感谢任何想法/帮助。

4 个答案:

答案 0 :(得分:14)

循环访问DataTable列并确定哪些是DateTime值。如果找到一个,则设置列的格式,如下所示。我还没有对代码进行测试,但它应该是遵循的一般逻辑。

...
wsDt.Cells["A1"].LoadFromDataTable(dt, true, TableStyles.None);
int colNumber = 1;

foreach (DataColumn col in dt.Columns) 
{        
    if (col.DataType == typeof(DateTime))
    { 
         wsDt.Column(colNumber).Style.Numberformat.Format = "MM/dd/yyyy hh:mm:ss AM/PM";
    }    
    colNumber++;      
}

wsDt.Cells[wsDt.Dimension.Address].AutoFitColumns();
Response.BinaryWrite(pck.GetAsByteArray());

答案 1 :(得分:3)

轻微修改上述答案。差不多正确。我用下面的代码修复了它。

wsDt.Cells["A1"].LoadFromDataTable(dt, true, TableStyles.None);

int colNumber = 0;
foreach (DataColumn col in dt.Columns) 
{
    colNumber++;
    if (col.DataType == typeof(DateTime))
    { 
       wsDt.Column(colNumber).Style.Numberformat.Format = "mm/dd/yyyy hh:mm:ss AM/PM"
    }          
}

wsDt.Cells[wsDt.Dimension.Address].AutoFitColumns();
Response.BinaryWrite(pck.GetAsByteArray());

答案 2 :(得分:3)

如果我错了,请纠正我,但我相信@desiguy和@Rick S的答案并不完全正确。

在@desiguy的回答中,语句colNumber++超出了数据表列的foreach循环。

因此,因为索引增量是在循环外完成的,所以循环将始终在colNumber=1上运行。

关于@Rick S的解决方案;如果我没有弄错,在EPPlus中,列索引从1开始而不是0,所以语句:

wsDt.Column(colNumber++).Style.Numberformat.Format = "mm/dd/yyyy hh:mm:ss AM/PM"

应如下:

wsDt.Column(++colNumber).Style.Numberformat.Format = "MM/dd/yyyy hh:mm:ss AM/PM"

虽然存在两个差异:

  1. colNumber++已更改为++colNumber。在这种情况下,列将从1开始而不是0,因此您不会收到错误

  2. mm/dd/yyyy已更改为MM/dd/yyyy,因为小写mm代表分钟而非月份,因此,MM是月份表示的正确语法< / p>

  3. 另一个更简单的解决方案是仅将int colNumber = 0;设置为int colNumber = 1;并获得相同的结果。

答案 3 :(得分:1)

完全更正(和测试)的代码:

        int colNumber = 0;

        foreach (DataColumn col in DataToExport.Columns)
        {
            colNumber++;
            if (col.DataType == typeof(DateTime))
            {
                wsDt.Column(colNumber).Style.Numberformat.Format = "MM/dd/yyyy hh:mm:ss AM/PM";
            }
        }