假如数据库表中某个字段存放的值“1”和“0”分别代表“是”和“否”,要在DataGridView中显示“是”和“否”,一般用两种方法,一种是在sql中直接判断获取,另一种是在DataGridView的CellFormatting事件中设置。
下面介绍的是第二种情况下的处理。
举个例子,DataGridView的第4列需要在金额后面加个“元”,在第14列根据1和0显示为相应的是和否,在显示的时候可以这样设置:

private void dgData_CellFormatting(object sender, DataGridViewCellFormattingEventArgs e)
{
if (e.ColumnIndex == 3)
{
if (e.Value != null)
e.Value = e.Value.ToString() + "元";
}
if (e.ColumnIndex == 13)
{
if (e.Value != null)
e.Value = e.Value.ToString() == "1" ? "是" : "否";
}
}

但是在循环行列导出Excel的时候并不会得到格式化后的数据,需要在循环列时判断是哪一列,再调用方法格式化列,如下:

private string FormatRMB(string s)
{
return s + "元";
}
private string FormatStatu(string s)
{
return s == "1" ? "已放款" : "未放款";
} private void ToExcel(int columnIndex, string fileName)
{
List<int> indexList = dgData.CheckBoxSelect(0);
if (indexList.Count == 0)
{
MessageBox.Show("请先选择!");
return;
}
using (SaveFileDialog saveFileDialog = new SaveFileDialog())
{
saveFileDialog.Title = "导出Excel文件到";
saveFileDialog.Filter = "Execl files(*.xls)|All Files(*.*)";
saveFileDialog.FileName = fileName;
saveFileDialog.AddExtension = true;
saveFileDialog.RestoreDirectory = true;
if (saveFileDialog.ShowDialog() == DialogResult.OK)
{
IWorkbook workbook = new HSSFWorkbook();
ISheet sheet = workbook.CreateSheet();
IRow rowTitle = sheet.CreateRow(0);
for (int j = columnIndex; j < dgData.Columns.Count; j++)
{
string headerText = dgData.Columns[j].HeaderText;
rowTitle.CreateCell(j - 1).SetCellValue(headerText);
}
for (int i = 0; i < indexList.Count; i++)
{
int selRowIndex = indexList[i];
IRow row = sheet.CreateRow(i + 1);
for (int j = columnIndex; j < dgData.Columns.Count; j++)
{
string val = dgData.Rows[selRowIndex].Cells[j].Value == null ? "" : dgData.Rows[selRowIndex].Cells[j].Value.ToString();
//格式化列数据
if (j == 3)
val = FormatRMB(val);
else if (j == 13)
val = FormatStatu(val); row.CreateCell(j - 1).SetCellValue(val);
}
}
FileStream outFs = new FileStream(saveFileDialog.FileName, FileMode.Create);
workbook.Write(outFs);
outFs.Close();
}
}
}

这里有一个缺点就是这个导出的方法失去了通用性,其它页面调用不了。
解决方法是在这个导出方法中再传一个字典参数,存放要格式化的列索引和委托,当循环到这一列时就执行这一个委托方法。
如下所示,现在已经是一个通用的方法,放在一个静态类中作为扩展方法,调用:

private void btnToExcel_Click(object sender, EventArgs e)
{
string fileName = string.Format("test_{0}.xls", DateTime.Now.ToString("yyyyMMddHHmmss"));
Dictionary<int, Func<string, string>> dict = new Dictionary<int, Func<string, string>>();
dict.Add(3, FormatRMB);
dict.Add(13, FormatStatu);
dgData.ExportExcel(1, fileName, dict);
}
//引用NPOI.dll
//using NPOI.SS.UserModel;
//using NPOI.HSSF.UserModel; public static class Extenstions
{
/// <summary>
/// Excel导出
/// </summary>
/// <param name="dgv">DataGridView控件ID</param>
/// <param name="columnIndex">从哪一列开始导出(因为有些列是checkbox)</param>
/// <param name="fileName">保存的文件名</param>
/// <param name="dict">存储列数据格式化的列号与方法字典</param>
public static void ExportExcel(this DataGridView dgv, int columnIndex, string fileName, Dictionary<int, Func<string, string>> dict)
{
List<int> indexList = dgv.CheckBoxSelect(0);
if (indexList.Count == 0)
{
MessageBox.Show("请先选择!");
return;
}
using (SaveFileDialog saveFileDialog = new SaveFileDialog())
{
saveFileDialog.Title = "导出Excel文件到";
saveFileDialog.Filter = "Execl files(*.xls)|All Files(*.*)";
saveFileDialog.FileName = fileName;
saveFileDialog.AddExtension = true;
saveFileDialog.RestoreDirectory = true;
if (saveFileDialog.ShowDialog() == DialogResult.OK)
{
IWorkbook workbook = new HSSFWorkbook();
ISheet sheet = workbook.CreateSheet();
IRow rowTitle = sheet.CreateRow(0);
for (int j = columnIndex; j < dgv.Columns.Count; j++)
{
string headerText = dgv.Columns[j].HeaderText;
rowTitle.CreateCell(j - 1).SetCellValue(headerText);
}
for (int i = 0; i < indexList.Count; i++)
{
int selRowIndex = indexList[i];
IRow row = sheet.CreateRow(i + 1);
for (int j = columnIndex; j < dgv.Columns.Count; j++)
{
string val = dgv.Rows[selRowIndex].Cells[j].Value == null ? "" : dgv.Rows[selRowIndex].Cells[j].Value.ToString();
//格式化列数据
if (dict.ContainsKey(j))
{
var v = dict.First(t => t.Key == j).Value;
val = v(val);
}
row.CreateCell(j - 1).SetCellValue(val);
}
}
FileStream outFs = new FileStream(saveFileDialog.FileName, FileMode.Create);
workbook.Write(outFs);
outFs.Close();
}
}
}
}

一个通用的DataGridView导出Excel扩展方法(支持列数据格式化)的更多相关文章

  1. asp.net中导出Excel的方法

    一.asp.net中导出Excel的方法: 本文转载 在asp.net中导出Excel有两种方法,一种是将导出的文件存放在服务器某个文件夹下面,然后将文件地址输出在浏览器上:一种是将文件直接将文件输出 ...

  2. ASP.NET导出excel表方法汇总

    asp.net里导出excel表方法汇总  1.由dataset生成 public void CreateExcel(DataSet ds,string typeid,string FileName) ...

  3. 懒人小工具:自动生成Model,Insert,Select,Delete以及导出Excel的方法

    在开发的过程中,我们为了节约时间,往往会将大量重复机械的代码封装,考虑代码的复用性,这样我们可以节约很多时间来做别的事情.最近跳槽到一节webform开发的公司,主要是开发自己公司用的ERP.开始因为 ...

  4. 懒人小工具:T4自动生成Model,Insert,Select,Delete以及导出Excel的方法

    之前写了篇文章,懒人小工具:[自动生成Model,Insert,Select,Delete以及导出Excel的方法](http://www.jianshu.com/p/d5b11589174a),但是 ...

  5. 懒人小工具:T4生成实体类Model,Insert,Select,Delete以及导出Excel的方法

    由于最近公司在用webform开发ERP,用到大量重复机械的代码,之前写了篇文章,懒人小工具:自动生成Model,Insert,Select,Delete以及导出Excel的方法,但是有人觉得这种方法 ...

  6. 懒人小工具1:winform自动生成Model,Insert,Select,Delete以及导出Excel的方法

       懒人小工具2:T4自动生成Model,Insert,Select,Delete以及导出Excel的方法    github地址:https://github.com/Jimmey-Jiang/J ...

  7. c# datagridview导出Excel文件 问题

    今天vs2010c#开发做datagridview导出Excel文件时,发现一个问题,和大家探讨一下: 第一种方式:写流的方式 private void button_Excel_Click(obje ...

  8. 从DataGridView导出Excel

    从DataGridView导出Excel的两种情况,不多说,直接记录代码(新建类,直接引用传入参数). using System; using System.Collections.Generic; ...

  9. Python学习随笔:使用xlwings设置和操作excel多行多列数据以及设置数据字体颜色填充色对齐方式的方法

    ☞ ░ 前往老猿Python博文目录 ░ 在前面老猿的文章中,<Python学习随笔:使用xlwings读取和操作Excel文件>.<Python学习随笔:使用xlwings读取和操 ...

随机推荐

  1. Socket请求和Http请求的各自特点、区别及适用场景

    Socket实现服务器与客户端之间的物理连接,并进行数据传输.主要有TCP/UDP两个协议.Socket处于网络协议的传输层.TCP:传输控制协议,面向连接的的协议,稳定可靠.当客户和服务器彼此交换数 ...

  2. 我的web框架设计

    做了很久的web开发,学了webform和mvc自己总结了,觉得当下的构架还是有改进的可能的. 其实首先说下我的一些认识(个人认知,欢迎讨论,谢绝砸砖). 我觉得对计算机和数据的操作,本身就是一个单向 ...

  3. python中的构造函数和析构函数

    python中的特殊方法,其中两个,构造函数和析构函数的作用: 比说“__init__”这个构造函数,具有初始化的作用,也就是当该类被实例化的时候就会执行该函数.那么我们就可以把要先初始化的属性放到这 ...

  4. Linux内核设计第四周——扒开系统调用三层皮

    Linux内核设计第四周 ——扒开系统调用三层皮 一.知识点总结 (一).系统调用基础知识 1.用户态和内核态 内核态:在高级别的状态下,代码可以执行特权指令,访问任意的物理地址: 用户态:在相应的低 ...

  5. [LeetCode]题解(python):120 Triangle

    题目来源 https://leetcode.com/problems/triangle/ Given a triangle, find the minimum path sum from top to ...

  6. Fiddler-010-网络延时应用小技巧-模拟低网速环境

    在日常的网络测试中,经常需要测试网络超时或在网络传输速率不佳的情况的应用场景,而与此同时我们有时手边资源有限,实现在各种真实网络(2G\3G)环境下测试有些局限性.其实 fiddler 已经提供了类似 ...

  7. Android 第三方开源下拉框:NiceSpinner

    Android原生的下拉框Spinner基本上可以满足Android开发对于下拉选项的设计需求,但现在越来越流行的下拉框不满足于Android原生提供的下拉框Spinner所提供的设计样式,而改用自定 ...

  8. PL/SQL 存储函数和过程

    --存储过程 .不带参: create or replace procedure 存储过程名 as|is --说明部分 begin --执行的语句: end: 调用存储过程 execute 存储过程名 ...

  9. sql server 中删除表中数据truncate和delete的区别(转载自.net学习网)

    我们都知道truncate table可以用来删除整个表的内容,它与delete后面不跟where条件的效果是一样.但除此之外,我们还清楚它们之间有其它的区别吗?本章我们将一起讨论truncate与d ...

  10. csuoj 1511: 残缺的棋盘

    http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1511 1511: 残缺的棋盘 时间限制: 1 Sec  内存限制: 128 MB 题目描述 输入 ...