寫一個通用的List集合導出excel的通用方法


前幾天要做一個數據導出Excel 

我就打算寫一個通用的。

這樣一來用的時候也方便,數據主要是通過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;
        }

  在寫這個的時候就遇到了一些問題。

  剛開始是打算用反射進去獲取,因為剛開始我自己試了一下(我手動創建了一個list集合里面的對象也是自己手動輸入的)

  這個時候用

foreach (System.Reflection.PropertyInfo p in tsc.GetType().GetProperties())
{
    row.CreateCell().SetCellValue(p.GetValue(tsc));
}

  這串代碼來往excel里面插入是沒有問題的。

  但是后來發現我自己創建的list和數據庫查詢之后返回的list不一樣。

  我數據庫框架用的dapper,接受集合的時候用的是List<dynamic>

  這時候就用反射獲取不到有多少個屬性了,也就取不到值了。

  后來我想既然這樣我就把他轉成json在把他轉成dynamic。

  后來試了一下,果然可以獲取的到屬性長度的數組。

  但是不能用foreach,因為這樣會出錯,給excel每一列賦值的時候需要傳索引號。

  

   我也就是我單獨把他拉出來的原因。

  

   但是這樣用循環依次獲取屬性的值會出問題,會報錯。

 

 

  

   然后我只得用這個方法了。用這樣的話,還需要自己定義一個數組把當前對象有字段的名稱告訴這個方法,所以略顯麻煩,所以之前一直在搞不要輸入的按照順序直接賦值的。但是沒弄出來:)

  之所以能用這個方法是因為把對象轉成json在把json轉成對象后這個對象是Jobject 就是Newtonsoft.Json里面的一個東西。他支持用【】來獲取數據

  dynamic是不支持【】獲取屬性的值的。

  還有就是SetCellValue不加Convert.ToString有時候會報錯,報具有二義性,我F12看了一下源碼,        

  

  應該是這兩個有點小差異,所以轉換一下就好了  

  

   與用法就是這個樣子的


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM