SpringBoot使用Apache POI庫(kù)讀取Excel文件的操作詳解
項(xiàng)目背景
假設(shè)我們需要開(kāi)發(fā)一個(gè)功能,讀取一個(gè)Excel文件中的數(shù)據(jù)并進(jìn)行處理。通常,這樣的需求會(huì)出現(xiàn)在以下場(chǎng)景中:
- 數(shù)據(jù)遷移:將Excel表格的數(shù)據(jù)導(dǎo)入數(shù)據(jù)庫(kù)。
- 數(shù)據(jù)分析:對(duì)Excel中的數(shù)據(jù)進(jìn)行匯總、統(tǒng)計(jì)分析。
- 批量處理:從Excel文件中讀取配置信息或參數(shù)進(jìn)行批量處理。 在本篇文章中,我們將展示如何使用Java讀取Excel文件,獲取其中的數(shù)據(jù),并展示如何將這些數(shù)據(jù)轉(zhuǎn)化為業(yè)務(wù)對(duì)象以便后續(xù)處理。
依賴導(dǎo)入
首先,你需要在項(xiàng)目中添加Apache POI的依賴。這里使用的是Apache POI 3.x版本,你可以在pom.xml中加入如下依賴:
<dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>5.2.3</version> </dependency>
這個(gè)依賴包包括了讀取xlsx格式的支持,如果需要支持更老的xls格式,可以再加上poi模塊。
讀取Excel模板的實(shí)現(xiàn)
接下來(lái),我們來(lái)看一個(gè)簡(jiǎn)單的示例代碼,展示如何讀取Excel文件的內(nèi)容,并對(duì)數(shù)據(jù)進(jìn)行處理。我們將以一個(gè)示例Excel表格為例,假設(shè)表格的內(nèi)容如下:
姓名 | 年齡 | 性別 |
---|---|---|
張三 | 25 | 男 |
李四 | 30 | 女 |
王五 | 28 | 男 |
代碼實(shí)現(xiàn)
import com.fasterxml.jackson.databind.ObjectMapper; import lombok.extern.slf4j.Slf4j; import org.apache.poi.ss.usermodel.*; import org.apache.poi.xssf.usermodel.XSSFWorkbook; import org.springframework.stereotype.Service; import java.io.FileInputStream; import java.io.IOException; import java.util.ArrayList; import java.util.List; /** * 讀取Excel文件并進(jìn)行數(shù)據(jù)處理的服務(wù)類 */ @Service @Slf4j public class ReadExcelServiceImpl { public Boolean readExcel() { try { String pathStr = "/path/to/your/excel/file.xlsx"; // excel文件路徑 FileInputStream fis = new FileInputStream(pathStr); // 創(chuàng)建一個(gè)工作簿對(duì)象 Workbook workbook = new XSSFWorkbook(fis); // 獲取第一個(gè)工作表 Sheet sheet = workbook.getSheetAt(0); // 獲取總行數(shù) int lastRowNum = sheet.getLastRowNum(); // 存放Excel讀取的數(shù)據(jù)列表 List<ExcelDemoInfoDTO> demoInfoList = new ArrayList<>(); // 讀取數(shù)據(jù)。循環(huán)遍歷行,從第二行開(kāi)始,假設(shè)第一行是標(biāo)題行 for (int i = 1; i <= lastRowNum; i++) { log.info("Reading row {}", i); Row row = sheet.getRow(i); if (row != null) { try { // 獲取單元格的值 String cell0 = getCellValue(row.getCell(0)); // 姓名 String cell1 = getCellValue(row.getCell(1)); // 年齡 String cell2 = getCellValue(row.getCell(2)); // 性別 // 創(chuàng)建數(shù)據(jù)對(duì)象并設(shè)置字段 ExcelDemoInfoDTO demoInfoDTO = new ExcelDemoInfoDTO(); demoInfoDTO.setName(cell0); demoInfoDTO.setAge(Integer.parseInt(cell1)); demoInfoDTO.setGender(cell2); // 將數(shù)據(jù)對(duì)象加入到列表 demoInfoList.add(demoInfoDTO); } catch (Exception e) { log.error("Error reading row {}", i, e); } } } // 使用Jackson將讀取的數(shù)據(jù)轉(zhuǎn)換為JSON字符串 ObjectMapper mapper = new ObjectMapper(); String json = mapper.writeValueAsString(demoInfoList); System.out.println(json); // 關(guān)閉資源 workbook.close(); fis.close(); } catch (IOException e) { e.printStackTrace(); return false; } return true; } /** * 獲取單元格的值,處理不同類型的單元格 * * @param cell 單元格對(duì)象 * @return 單元格的字符串值 */ private static String getCellValue(Cell cell) { if (cell == null) { return ""; } switch (cell.getCellType()) { case STRING: return cell.getStringCellValue(); case NUMERIC: return String.valueOf((int) cell.getNumericCellValue()); default: return ""; } } }
代碼解析
- 打開(kāi)Excel文件 使用FileInputStream打開(kāi)指定路徑的Excel文件,然后通過(guò)XSSFWorkbook將其加載為工作簿(Workbook)。
- 讀取Excel工作表 通過(guò)workbook.getSheetAt(0)獲取第一個(gè)工作表(Sheet)。你可以根據(jù)需要更改getSheetAt中的索引值來(lái)獲取其他工作表。
- 遍歷行和列 使用sheet.getRow(i)獲取每一行的數(shù)據(jù)。我們從第二行開(kāi)始讀取(i=1),因?yàn)榈谝恍型ǔJ菢?biāo)題行。
- 獲取單元格內(nèi)容 通過(guò)row.getCell(i)獲取每一列的內(nèi)容,并使用getCellValue方法根據(jù)單元格的類型(字符串、數(shù)字等)獲取對(duì)應(yīng)的值。
- 封裝數(shù)據(jù) 將每行的數(shù)據(jù)封裝為一個(gè)業(yè)務(wù)對(duì)象(ExcelDemoInfoDTO),并將其加入到一個(gè)列表中。
- 轉(zhuǎn)換為JSON格式 使用Jackson庫(kù)將讀取的數(shù)據(jù)轉(zhuǎn)化為JSON格式,以便后續(xù)的處理或傳輸。
- 資源關(guān)閉 使用完畢后,關(guān)閉workbook和FileInputStream以釋放資源。
ExcelDemoInfoDTO 數(shù)據(jù)傳輸對(duì)象
為了更好地封裝數(shù)據(jù),我們創(chuàng)建一個(gè)簡(jiǎn)單的DTO(數(shù)據(jù)傳輸對(duì)象)類ExcelDemoInfoDTO:
public class ExcelDemoInfoDTO { private String name; private int age; private String gender; // Getters and Setters public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getGender() { return gender; } public void setGender(String gender) { this.gender = gender; } }
總結(jié)
在本次實(shí)踐中,我們介紹了如何使用Apache POI庫(kù)
在Java中讀取Excel文件,獲取其中的數(shù)據(jù),并將這些數(shù)據(jù)封裝為業(yè)務(wù)對(duì)象。通過(guò)這種方式,我們可以靈活地讀取各種格式的Excel數(shù)據(jù),并進(jìn)行后續(xù)的業(yè)務(wù)處理。對(duì)于更復(fù)雜的Excel文件,我們還可以進(jìn)一步擴(kuò)展代碼來(lái)處理更多類型的單元格、跨工作表讀取等情況。
以上就是SpringBoot使用Apache POI庫(kù)讀取Excel文件的操作詳解的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot Apache POI庫(kù)讀取Excel的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java使用任務(wù)架構(gòu)執(zhí)行任務(wù)調(diào)度示例
在Java 5.0之前啟動(dòng)一個(gè)任務(wù)是通過(guò)調(diào)用Thread類的start()方法來(lái)實(shí)現(xiàn)的,5.0里提供了一個(gè)新的任務(wù)執(zhí)行架構(gòu)使你可以輕松地調(diào)度和控制任務(wù)的執(zhí)行,并且可以建立一個(gè)類似數(shù)據(jù)庫(kù)連接池的線程池來(lái)執(zhí)行任務(wù),下面看一個(gè)示例2014-01-01Java設(shè)計(jì)模式七大原則之里氏替換原則詳解
在面向?qū)ο蟮某绦蛟O(shè)計(jì)中,里氏替換原則(Liskov Substitution principle)是對(duì)子類型的特別定義。本文將為大家詳細(xì)介紹Java設(shè)計(jì)模式七大原則之一的里氏替換原則,需要的可以參考一下2022-02-02springboot實(shí)現(xiàn)郵箱發(fā)送(激活碼)功能的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用springboot實(shí)現(xiàn)郵箱發(fā)送(激活碼)功能,文中的示例代碼簡(jiǎn)潔易懂,有需要的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-10-10MyBatis框架迭代器模式實(shí)現(xiàn)原理解析
這篇文章主要介紹了MyBatis框架迭代器模式實(shí)現(xiàn)原理解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03Mybatis千萬(wàn)級(jí)數(shù)據(jù)查詢的解決方式,避免OOM問(wèn)題
這篇文章主要介紹了Mybatis千萬(wàn)級(jí)數(shù)據(jù)查詢的解決方式,避免OOM問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01Java中BM(Boyer-Moore)算法的圖解與實(shí)現(xiàn)
本文主要介紹了兩個(gè)大的部分,第一部分通過(guò)圖解的方式講解BM算法,第二部分則代碼實(shí)現(xiàn)一個(gè)簡(jiǎn)易的BM算法,感興趣的小伙伴可以學(xué)習(xí)一下2022-05-05基于java下載中g(shù)etContentLength()一直為-1的一些思路
下面小編就為大家?guī)?lái)一篇基于java下載中g(shù)etContentLength()一直為-1的一些思路。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-06-06