• [转]asp.net导出Excel/Csv格式数据最优方案(C#)


    本文转自:http://www.cnblogs.com/lovenets2008/archive/2008/05/06/1184234.html

    原文如下:

    好久没有写点什么了,也许是太忙。一年了,积累了不少好的东东,有机会时就写出来与大家分享。

    好,言归正传。
    导出到Excel/Csc文件并不难,所以就有好多方法:控件直接Render、把DataSet输出成String再Write出来等,(当然如果调用 Excel程序的库文件的话还可以使用更强的直接操作Excel的方法,但这种方法用于Web服务显得有点要求太高:必须让Web服务器安装指定版本的 Excel或其支持库文件)。就其前两种方法,实际上也是一样的,Render也是把由DataSet转变的View生成为一个Table输出到客户端而 已,只不过隐藏了细节,如果不信,你用EditPlus什么的看看生成的.xls文件就知道了。

    Excel 的识别力太强了,以至于它本身的格式、Csv格式、Tab分隔符格式、网页的Table格式等都能够很好的打开。但是它“太聪明”了,以至于自动识别数字 和字符串,而且要把超过11位的数字自动变为科学计数法的格式,你试试输入“123456789012”,离开那个单元格,就成“123457E+11” 了,够聪明的吧,不过有时会让我们感觉不便,因为我输入的就是我自己的身份证号码,尾巴上没有“X”,本来好好的15位数字,得现在成这么个计数法了。那 我就改改显示格式吧,改为把数字显示为文本,好了。可是国家的身份证升级了,号码变成18位,我把它输入到数据库,导出来时,用刚才的方法处理过,18位 没错,可是最后三位怎么都是零了!Excel为我们做了太多的事,不管是应该的还是不应该的。

    怎么解决?请看代码:

        public static string ExportTable(DataSet ds)
        {
            string data = "";
            //data = ds.DataSetName + "\n";

            foreach (DataTable tb in ds.Tables)
            {
                //data += tb.TableName + "\n";
                data += "<table cellspacing=\"0\" cellpadding=\"5\" rules=\"all\" border=\"1\">";
                //写出列名
                data += "<tr style=\"font-weight: bold; white-space: nowrap;\">";
                foreach (DataColumn column in tb.Columns)
                {
                    data += "<td>" + column.ColumnName + "</td>";
                }
                data += "</tr>";

                //写出数据
                foreach (DataRow row in tb.Rows)
                {
                    data += "<tr>";
                    foreach (DataColumn column in tb.Columns)
                    {
                        if (column.ColumnName.Equals("证件编号") || column.ColumnName.Equals("报名编号"))
                            data += "<td style=\"vnd.ms-excel.numberformat:@\">" + row[column].ToString() + "</td>";
                        else
                            data += "<td>" + row[column].ToString() + "</td>";
                    }
                    data += "</tr>";
                }
                data += "</table>";
            }

            return data;
        }


        public static void ExportDsToXls(Page page, string sql)
        {
            ExportDsToXls(page, "FileName", sql);
        }
        public static void ExportDsToXls(Page page, string fileName, string sql)
        {
            DataSet ds = DBUtil.GetDataSet(sql);
            if (ds != null) ExportDsToXls(page, fileName, ds);
        }
        public static void ExportDsToXls(Page page, DataSet ds)
        {
            ExportDsToXls(page, "FileName", ds);
        }
        public static void ExportDsToXls(Page page, string fileName, DataSet ds)
        {
            page.Response.Clear();
            page.Response.Buffer = true;
            page.Response.Charset = "GB2312";
            //page.Response.Charset = "UTF-8";
            page.Response.AppendHeader("Content-Disposition", "attachment;filename=" + fileName + System.DateTime.Now.ToString("_yyMMdd_hhmm") + ".xls");
            page.Response.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");//设置输出流为简体中文
            page.Response.ContentType = "application/ms-excel";//设置输出文件类型为excel文件。
            page.EnableViewState = false;
            page.Response.Write(ExportTable(ds));
            page.Response.End();
        }
    //style="vnd.ms-excel.numberformat:@" 可以去除自动科学计数法的困扰
    //输出为Table,能够最大限度的减少字段中数据对生成的文件格式的影响,在这里我没有处理数据中含有HTML标签的情况 在页面后台中,这样使用就可以了:

        protected void lbtnToExcel_Click(object sender, EventArgs e)
        {
            string strWhere = BuildSearchWhereString();
            string strOrder = this.hidOrderString.Value;
            string sql = "SELECT 报名编号, 证件编号, 姓名, 考区考点, 报考类别, "
                + "行政区划名称 AS 行政区划, 单位名称 AS 工作单位, 毕业学校名称, 毕业专业名称 AS 毕业专业, 毕业年月, "
                + "通讯地址, 性别"
                + " from [VW报名]";
            if (!string.IsNullOrEmpty(strWhere)) sql += " where " + strWhere;
            if (!string.IsNullOrEmpty(strOrder)) sql += " order by " + strOrder;
            else sql += " order by [报考类别]";
            PageExport.ExportDsToXls(this.Page, "BaoMing", sql);
            dataBind();
        }

     暂写到这里,休息。
    更新日期:2008-5-5

    #7楼 [楼主] 2008-05-09 12:33 爱网2008    

    修改了这个函数:在导出1k条数据时,估计速度提高上百倍,数据量越大越明显,原理很简单,StringBuilder的性能和“+”的性能的区别。
    public static string ExportTable(DataSet ds)
    {
    StringBuilder sb = new StringBuilder();
    //data = ds.DataSetName + "\n";
    int count = 0;

    foreach (DataTable tb in ds.Tables)
    {
    //data += tb.TableName + "\n";
    sb.AppendLine("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=gb2312\">");
    sb.AppendLine("<table cellspacing=\"0\" cellpadding=\"5\" rules=\"all\" border=\"1\">");
    //写出列名
    sb.AppendLine("<tr style=\"font-weight: bold; white-space: nowrap;\">");
    foreach (DataColumn column in tb.Columns)
    {
    sb.AppendLine("<td>" + column.ColumnName + "</td>");
    }
    sb.AppendLine("</tr>");

    //写出数据
    foreach (DataRow row in tb.Rows)
    {
    sb.Append("<tr>");
    foreach (DataColumn column in tb.Columns)
    {
    if (column.ColumnName.Equals("证件编号") || column.ColumnName.Equals("报名编号"))
    sb.Append("<td style=\"vnd.ms-excel.numberformat:@\">" + row[column].ToString() + "</td>");
    else
    sb.Append("<td>" + row[column].ToString() + "</td>");
    }
    sb.AppendLine("</tr>");
    count++;
    }
    sb.AppendLine("</table>");
    }

    return sb.ToString();
    }

    稍后会提供下载版

  • 相关阅读:
    POJ 1141 Brackets Sequence (区间DP)
    UVaLive 6585 && Gym 100299F Draughts (暴力+回溯)
    UVaLive 6950 && Gym 100299K Digraphs (DFS找环或者是找最长链)
    UVaLive 6588 && Gym 100299I (贪心+构造)
    UVa 1611 Crane (构造+贪心)
    Gym 100299C && UVaLive 6582 Magical GCD (暴力+数论)
    UVa 1642 Magical GCD (暴力+数论)
    UVaLive 6591 && Gym 100299L Bus (水题)
    UVaLive 6581 && Gym 100299B What does the fox say? (模拟+STL)
    HDU 5898 odd-even number (数位DP)
  • 原文地址:https://www.cnblogs.com/blsong/p/1621892.html
Copyright © 2020-2023  润新知