excel操作之poi-ooxml


  目前市場上流行的對於excel處理的框架大致有兩種:poi和jxl。對於這兩種框架,我們可以做一個簡單的對比:
   1 開發團隊:poi是Apache旗下的一個開源項目,由Apache官方維護,jxl好像是一個個人維護的開源項目。
   2 各自優點:poi對公式支持較好,jxl不算好 。jxl提供對圖片的支持(僅僅PNG格式),poi支持。(就這一條來看財務軟件就該選poi,而媒體類的軟件就該選jxl了)
   3 內存消耗:由於jxl在對資源回收利用方面做了相當的功課,在內存消耗上jxl是略勝於poi的。所以對於大數據量的軟件導入來說,選擇jxl是比較合算的,當然數據量小的基本沒有差別。
   4 運行速度: 估計是內存消耗多的緣故,poi對於讀寫速度這一功能做的好像比jxl好了不少,並且支持壓縮excel。
    對比了這麼多,對於自己項目該使用哪個框架,應該也十分明顯了(當然這些也都是從網上搜集來的,不保證有錯誤的地方)。
    這裏我記錄一下poi的使用吧,都挺簡單的,基本原理都是將excel表格數據提取出來組成一個list。然後對應這個list自己去做循環對應自己數據表的數據就行了。需要說明的一點是如果是數字類型的話,讀出來的數據一般都是以double類型返回給你的,比如你在excel裏面寫的是100,讀取出來的數據就是100.0.這點比較煩人,當然自己做一下處理就好了。
    還有一點就是poi也有兩個不同的jar包,分別是處理excel2003和excel2007+的,對應的是poi和poi-ooxml。畢竟poi-ooxml是poi的升級版本,處理的單頁數據量也是百萬級別的,所以我們選擇的也是poi-ooxml。好了,下面就貼上代碼吧,註釋較多,就不多做囉嗦了。,前提是引入包:

   以下代碼完全可以作爲一個excel工具類遷移到自己的項目中:
[java] view plain copy
  1. /** 
  2.  * 處理excel讀入的工具類 
  3.  * Created by Liujishuai on 2015/8/5. 
  4.  */  
  5. public class ExcelUtils {  
  6.     /** 
  7.      * 要求excel版本在2007以上 
  8.      * 
  9.      * @param file 文件信息 
  10.      * @return 
  11.      * @throws Exception 
  12.      */  
  13.     public static List<List<Object>> readExcel(File file) throws Exception {  
  14.         if(!file.exists()){  
  15.             throw new Exception("找不到文件");  
  16.         }  
  17.         List<List<Object>> list = new LinkedList<List<Object>>();  
  18.         XSSFWorkbook xwb = new XSSFWorkbook(new FileInputStream(file));  
  19.         // 讀取第一張表格內容  
  20.         XSSFSheet sheet = xwb.getSheetAt(0);  
  21.         XSSFRow row = null;  
  22.         XSSFCell cell = null;  
  23.         for (int i = (sheet.getFirstRowNum() + 1); i <= (sheet.getPhysicalNumberOfRows() - 1); i++) {  
  24.             row = sheet.getRow(i);  
  25.             if (row == null) {  
  26.                 continue;  
  27.             }  
  28.             List<Object> linked = new LinkedList<Object>();  
  29.             for (int j = row.getFirstCellNum(); j <= row.getLastCellNum(); j++) {  
  30.                 Object value = null;  
  31.                 cell = row.getCell(j);  
  32.                 if (cell == null) {  
  33.                     continue;  
  34.                 }  
  35.                 switch (cell.getCellType()) {  
  36.                     case XSSFCell.CELL_TYPE_STRING:  
  37.                         //String類型返回String數據  
  38.                         value = cell.getStringCellValue();  
  39.                         break;  
  40.                     case XSSFCell.CELL_TYPE_NUMERIC:  
  41.                         //日期數據返回LONG類型的時間戳  
  42.                         if ("yyyy\"年\"m\"月\"d\"日\";@".equals(cell.getCellStyle().getDataFormatString())) {  
  43.                             //System.out.println(cell.getNumericCellValue()+":日期格式:"+cell.getCellStyle().getDataFormatString());  
  44.                             value = DateUtils.getMillis(HSSFDateUtil.getJavaDate(cell.getNumericCellValue())) / 1000;  
  45.                         } else {  
  46.                             //數值類型返回double類型的數字  
  47.                             //System.out.println(cell.getNumericCellValue()+":格式:"+cell.getCellStyle().getDataFormatString());  
  48.                             value = cell.getNumericCellValue();  
  49.                         }  
  50.                         break;  
  51.                     case XSSFCell.CELL_TYPE_BOOLEAN:  
  52.                         //布爾類型  
  53.                         value = cell.getBooleanCellValue();  
  54.                         break;  
  55.                     case XSSFCell.CELL_TYPE_BLANK:  
  56.                         //空單元格  
  57.                         break;  
  58.                     default:  
  59.                         value = cell.toString();  
  60.                 }  
  61.                 if (value != null && !value.equals("")) {  
  62.                     //單元格不爲空,則加入列表  
  63.                     linked.add(value);  
  64.                 }  
  65.             }  
  66.             if (linked.size()!= 0) {  
  67.                 list.add(linked);  
  68.             }  
  69.         }  
  70.         return list;  
  71.     }  
  72.     /** 
  73.      * 要求excel版本在2007以上 
  74.      * 
  75.      * @param fileInputStream 文件信息 
  76.      * @return 
  77.      * @throws Exception 
  78.      */  
  79.     public static List<List<Object>> readExcel(FileInputStream fileInputStream) throws Exception {  
  80.         List<List<Object>> list = new LinkedList<List<Object>>();  
  81.         XSSFWorkbook xwb = new XSSFWorkbook(fileInputStream);  
  82.         // 讀取第一張表格內容  
  83.         XSSFSheet sheet = xwb.getSheetAt(1);  
  84.         XSSFRow row = null;  
  85.         XSSFCell cell = null;  
  86.         for (int i = (sheet.getFirstRowNum() + 1); i <= (sheet.getPhysicalNumberOfRows() - 1); i++) {  
  87.             row = sheet.getRow(i);  
  88.             if (row == null) {  
  89.                 continue;  
  90.             }  
  91.             List<Object> linked = new LinkedList<Object>();  
  92.             for (int j = row.getFirstCellNum(); j <= row.getLastCellNum(); j++) {  
  93.                 Object value = null;  
  94.                 cell = row.getCell(j);  
  95.                 if (cell == null) {  
  96.                     continue;  
  97.                 }  
  98.                 switch (cell.getCellType()) {  
  99.                     case XSSFCell.CELL_TYPE_STRING:  
  100.                         value = cell.getStringCellValue();  
  101.                         break;  
  102.                     case XSSFCell.CELL_TYPE_NUMERIC:  
  103.                         if ("yyyy\"年\"m\"月\"d\"日\";@".equals(cell.getCellStyle().getDataFormatString())) {  
  104.                             //System.out.println(cell.getNumericCellValue()+":日期格式:"+cell.getCellStyle().getDataFormatString());  
  105.                             value = DateUtils.getMillis(HSSFDateUtil.getJavaDate(cell.getNumericCellValue())) / 1000;  
  106.                         } else {  
  107.                             //System.out.println(cell.getNumericCellValue()+":格式:"+cell.getCellStyle().getDataFormatString());  
  108.                             value = cell.getNumericCellValue();  
  109.                         }  
  110.                         break;  
  111.                     case XSSFCell.CELL_TYPE_BOOLEAN:  
  112.                         value = cell.getBooleanCellValue();  
  113.                         break;  
  114.                     case XSSFCell.CELL_TYPE_BLANK:  
  115.                         break;  
  116.                     default:  
  117.                         value = cell.toString();  
  118.                 }  
  119.                 if (value != null && !value.equals("")) {  
  120.                     //單元格不爲空,則加入列表  
  121.                     linked.add(value);  
  122.                 }  
  123.             }  
  124.             if (linked.size()!= 0) {  
  125.                 list.add(linked);  
  126.             }  
  127.         }  
  128.         return list;  
  129.     }  
  130.   
  131.     /** 
  132.      * 導出excel 
  133.      * @param excel_name 導出的excel路徑(需要帶.xlsx) 
  134.      * @param headList  excel的標題備註名稱 
  135.      * @param fieldList excel的標題字段(與數據中map中鍵值對應) 
  136.      * @param dataList  excel數據 
  137.      * @throws Exception 
  138.      */  
  139.     public static void createExcel(String excel_name, String[] headList,  
  140.                                    String[] fieldList, List<Map<String, Object>> dataList)  
  141.             throws Exception {  
  142.         // 創建新的Excel 工作簿  
  143.         XSSFWorkbook workbook = new XSSFWorkbook();  
  144.         // 在Excel工作簿中建一工作表,其名爲缺省值  
  145.         XSSFSheet sheet = workbook.createSheet();  
  146.         // 在索引0的位置創建行(最頂端的行)  
  147.         XSSFRow row = sheet.createRow(0);  
  148.         // 設置excel頭(第一行)的頭名稱  
  149.         for (int i = 0; i < headList.length; i++) {  
  150.   
  151.             // 在索引0的位置創建單元格(左上端)  
  152.             XSSFCell cell = row.createCell(i);  
  153.             // 定義單元格爲字符串類型  
  154.             cell.setCellType(XSSFCell.CELL_TYPE_STRING);  
  155.             // 在單元格中輸入一些內容  
  156.             cell.setCellValue(headList[i]);  
  157.         }  
  158.         // ===============================================================  
  159.         //添加數據  
  160.         for (int n = 0; n < dataList.size(); n++) {  
  161.             // 在索引1的位置創建行(最頂端的行)  
  162.             XSSFRow row_value = sheet.createRow(n + 1);  
  163.             Map<String, Object> dataMap = dataList.get(n);  
  164.             // ===============================================================  
  165.             for (int i = 0; i < fieldList.length; i++) {  
  166.   
  167.                 // 在索引0的位置創建單元格(左上端)  
  168.                 XSSFCell cell = row_value.createCell(i);  
  169.                 // 定義單元格爲字符串類型  
  170.                 cell.setCellType(XSSFCell.CELL_TYPE_STRING);  
  171.                 // 在單元格中輸入一些內容  
  172.                 cell.setCellValue((dataMap.get(fieldList[i])).toString());  
  173.             }  
  174.             // ===============================================================  
  175.         }  
  176.         // 新建一輸出文件流  
  177.         FileOutputStream fos = new FileOutputStream(excel_name);  
  178.         // 把相應的Excel 工作簿存盤  
  179.         workbook.write(fos);  
  180.         fos.flush();  
  181.         // 操作結束,關閉文件  
  182.         fos.close();  
  183.     }  
  184. }  

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章