前幾天要作一個數據導出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看了一下源碼,
應該是這兩個有點小差別,因此轉換一下就行了
與用法就是這個樣子的