前几天要作一个数据导出Excel 数据库
我就打算写一个通用的。json
这样一来用的时候也方便,数据主要是经过Orm取的List。这样写一个通用的恰好。数组
public static void ListToExcel(List<dynamic> ts, string[] RowName, string[] ListCorrespondRow, bool IsRowName = false) { //建立工做簿对象 IWorkbook workbook = new HSSFWorkbook(); //建立工做表 ISheet sheet = workbook.CreateSheet("onesheet"); IRow row0 = sheet.CreateRow(0); for (int i = 0; i < RowName.Length; i++) { row0.CreateCell(i).SetCellValue(RowName[i]); } for (int r = 1; r <= ts.Count; r++) { //建立行row IRow row = sheet.CreateRow(r); dynamic tsc = ts[r-1];string sJson = JsonConvert.SerializeObject(tsc); dynamic sObj = JsonConvert.DeserializeObject<dynamic>(sJson); var sObjLen = sObj.GetType().GetProperties(); for (int j = 0; j < ListCorrespondRow.Length; j++) { //经过【】来取值 但必需要经过json转成的对象才能够这样取 row.CreateCell(j).SetCellValue(Convert.ToString(sObj[ListCorrespondRow[j]])); } //foreach (System.Reflection.PropertyInfo p in tsc.GetType().GetProperties()) //{ // //row.CreateCell().SetCellValue(p.GetValue(tsc)); //} //for (int j = 0; j < sObjLen.Length; j++) //{ //利用反射将对象里面的值添加到excel里面 添加的顺序是按照对象里面字段的顺序 注意和列名保持一致 // row.CreateCell(j).SetCellValue(sObjLen[j].GetValue(tsc)); //} } //建立流对象并设置存储Excel文件的路径 using (FileStream url = File.OpenWrite(@"C:/Users/13002/source/repos/练习/练习/WordDot/test3.xls")) { //导出Excel文件 workbook.Write(url); //Response.Write("<script>alert('写入成功!')</script>"); }; ////workbook.Write(); ////建立文件流 //MemoryStream bookStream = new MemoryStream(); ////文件写入流(向流中写入字节序列) //workbook.Write(bookStream); ////输出以前调用Seek(偏移量,游标位置) 把0位置指定为开始位置 //bookStream.Seek(0, SeekOrigin.Begin); //return bookStream; }
在写这个的时候就遇到了一些问题。app
刚开始是打算用反射进去获取,由于刚开始我本身试了一下(我手动建立了一个list集合里面的对象也是本身手动输入的)框架
这个时候用url
foreach (System.Reflection.PropertyInfo p in tsc.GetType().GetProperties()) { row.CreateCell().SetCellValue(p.GetValue(tsc)); }
这串代码来往excel里面插入是没有问题的。spa
可是后来发现我本身建立的list和数据库查询以后返回的list不同。excel
我数据库框架用的dapper,接受集合的时候用的是List<dynamic>code
这时候就用反射获取不到有多少个属性了,也就取不到值了。对象
后来我想既然这样我就把他转成json在把他转成dynamic。
后来试了一下,果真能够获取的到属性长度的数组。
可是不能用foreach,由于这样会出错,给excel每一列赋值的时候须要传索引号。
我也就是我单独把他拉出来的缘由。
可是这样用循环依次获取属性的值会出问题,会报错。
而后我只得用这个方法了。用这样的话,还须要本身定义一个数组把当前对象有字段的名称告诉这个方法,因此略显麻烦,因此以前一直在搞不要输入的按照顺序直接赋值的。可是没弄出来:)
之因此能用这个方法是由于把对象转成json在把json转成对象后这个对象是Jobject 就是Newtonsoft.Json里面的一个东西。他支持用【】来获取数据
dynamic是不支持【】获取属性的值的。
还有就是SetCellValue不加Convert.ToString有时候会报错,报具备二义性,我F12看了一下源码,
应该是这两个有点小差别,因此转换一下就行了
与用法就是这个样子的