java如何讀取Excel簡(jiǎn)單模板
場(chǎng)景:對(duì)于經(jīng)常需要導(dǎo)入Excel模板或數(shù)據(jù)來解析后加以應(yīng)用的,使用頻率非常之高,做了一個(gè)比較穩(wěn)定的版本,體現(xiàn)在這些地方
工具:org.apache.poi
使用前必須了解這些:
1、要解析,那肯定先判斷是不是Excel
2、xls后綴的Excel,是03版及以前的用HSSFWorkbook類
xlsx后綴的Excel,是07版及以后的用XSSFWorkbook解析
3、getWorkbook這個(gè)方法是我自己亂造各種Excel數(shù)據(jù)不斷測(cè)試搜索修正得出的結(jié)果,其他的像簡(jiǎn)單的判斷后綴xls還是xlsx來決定用HSSH還是XSSF是不保險(xiǎn)的,比如你可能沒遇過org.apache.poi.openxml4j.exceptions.InvalidFormatException這樣的異常,當(dāng)然這個(gè)異常仍然是因?yàn)镋xcel類型導(dǎo)致獲取workbook時(shí)出錯(cuò),然而我查到的結(jié)果是,Excel最底層是xml實(shí)現(xiàn)的,類型問題出在這兒,看異常的描述也可以稍微看出來openxml4j.exceptions
4 、可能出現(xiàn)空行,空的單元格,或者單元格值為空的情況,這些情況,在我的readExcel()方法里都考慮到了,為什么我不用迭代器,或者加強(qiáng)的for each循環(huán)?就是因?yàn)檫@些坑爹的空單元格或者空行啊,迭代器內(nèi)部在取cell單元格對(duì)象時(shí)跳過這些空的對(duì)象,who knows why?我也不知道,反正我測(cè)試過,跳過去了,本來5個(gè)單元格,一個(gè)空的,結(jié)果就只得到4個(gè)數(shù)據(jù),即使用cell.isEmpty()和cell!=null來判斷,也沒卵用,因?yàn)楸闅v的時(shí)候直接跳過去了,都沒有判斷的機(jī)會(huì)
5、取單元格數(shù)據(jù),這個(gè)就比較簡(jiǎn)單了,判斷單元格類型,根據(jù)類型做相應(yīng)的處理取出來,但是我覺得我這個(gè)getCellValue()的方法應(yīng)該有漏洞,先這么用著
下面上代碼,簡(jiǎn)單描述下關(guān)鍵部位
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.PushbackInputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
import java.util.List;
import org.apache.poi.POIXMLDocument;
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
import org.apache.poi.openxml4j.opc.OPCPackage;
import org.apache.poi.poifs.filesystem.POIFSFileSystem;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.xmlbeans.impl.piccolo.io.FileFormatException;
/**
*yanbiao 2016.10.25
*/
public class ExcelUtil {
private static final String EXTENSION_XLS = "xls";
private static final String EXTENSION_XLSX = "xlsx";
/**
* 文件檢查
*/
private void preReadCheck(String filePath) throws FileNotFoundException, FileFormatException {
File file = new File(filePath);
if (!file.exists()) {
throw new FileNotFoundException("導(dǎo)入的文件不存在:" + filePath);
}
if (!(filePath.endsWith(EXTENSION_XLS) || filePath.endsWith(EXTENSION_XLSX))) {
throw new FileFormatException("傳入的文件不是excel");
}
}
/**
* 取得WorkBook對(duì)象
* xls:HSSFWorkbook,03版
* xlsx:XSSFWorkbook,07版
*/
private Workbook getWorkbook(String filePath) throws IOException, InvalidFormatException {
//直接判斷后綴來返回相應(yīng)的Workbook對(duì)象多數(shù)情況沒問題,但是這個(gè)更保險(xiǎn),第3條已經(jīng)說明
Workbook wb = null;
InputStream is = new FileInputStream(filePath);
if (!is.markSupported()) {
is = new PushbackInputStream(is, 8);
}
if (POIFSFileSystem.hasPOIFSHeader(is)) {
return new HSSFWorkbook(is);
}
if (POIXMLDocument.hasOOXMLHeader(is)) {
return new XSSFWorkbook(OPCPackage.open(is));
}
throw new IllegalArgumentException("您的excel版本目前不支持poi解析");
}
/**
* 讀取excel文件內(nèi)容
*/
public Map<Integer, List<String>> readExcel(String filePath) throws FileNotFoundException, FileFormatException {
// 檢查和獲取workbook對(duì)象
this.preReadCheck(filePath);
Workbook wb = null;
Map<Integer,List<String>> map = new HashMap<Integer, List<String>>();
try {
wb = this.getWorkbook(filePath);
// 默認(rèn)只讀取第一個(gè)sheet
Sheet sheet = wb.getSheetAt(0);
int rowcount = sheet.getLastRowNum();//邏輯行,包括空行
int cellcount = sheet.getRow(0).getLastCellNum();//第一行(將來作為字段的行)有多少個(gè)單元格
for (int i=0;i<rowcount;i++) { //這里用最原始的for循環(huán)來保證每行都會(huì)被讀取
List<String> list = new ArrayList<String>();
Row row = sheet.getRow(i);
if(null!=row){
for (int j=0;j<cellcount;j++) {
list.add(getCellValue(row.getCell(j))); //這里也是用for循環(huán),用Cell c:row這樣的遍歷,空單元格就被拋棄了
}
System.out.println("第"+(row.getRowNum()+1)+"行數(shù)據(jù):"+list.toString());
map.put(row.getRowNum(), list);
}else{
for (int j=0;j<cellcount;j++) {
list.add("無數(shù)據(jù)");
}
System.out.println("第"+(i+1)+"行數(shù)據(jù):"+list.toString());
map.put(i, list);
}
}
} catch (Exception e) {
System.out.println("讀取Excel異常:"+e.getMessage());
e.printStackTrace();
} finally {
if (wb != null) {
try {
wb.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return map;
}
/**
* 取單元格的值
*/
private String getCellValue(Cell c) {
if (c == null) {
return "無數(shù)據(jù)";
}
String value = "";
switch (c.getCellType()){
case HSSFCell.CELL_TYPE_NUMERIC://數(shù)字
value = c.getNumericCellValue()+"";
break;
case HSSFCell.CELL_TYPE_STRING://字符串
value = c.getStringCellValue();
break;
case HSSFCell.CELL_TYPE_BOOLEAN://boolean
value = c.getBooleanCellValue()+"";
break;
case HSSFCell.CELL_TYPE_FORMULA://公式
value = c.getCellFormula()+"";
break;
case HSSFCell.CELL_TYPE_BLANK://空值
value= "無數(shù)據(jù)";
break;
case HSSFCell.CELL_TYPE_ERROR:
value = "非法字符";
break;
default:
value= "未知類型";
break;
}
return value;
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot添加jvm監(jiān)控實(shí)現(xiàn)數(shù)據(jù)可視化
這篇文章主要介紹了Springboot添加jvm監(jiān)控實(shí)現(xiàn)數(shù)據(jù)可視化,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
Java找不到或無法加載主類及編碼錯(cuò)誤問題的解決方案
今天小編就為大家分享一篇關(guān)于Java找不到或無法加載主類及編碼錯(cuò)誤問題的解決方案,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-02-02
Spring?Validation接口入?yún)⑿r?yàn)示例代碼
Spring?Validation是一種用于實(shí)現(xiàn)數(shù)據(jù)校驗(yàn)的框架,它提供了一系列的校驗(yàn)器,針對(duì)不同的數(shù)據(jù)類型可以使用不同的校驗(yàn)器進(jìn)行校驗(yàn),下面這篇文章主要給大家介紹了關(guān)于Spring?Validation接口入?yún)⑿r?yàn)的相關(guān)資料,需要的朋友可以參考下2023-06-06
用Spring Native將SpringBoot程序轉(zhuǎn)換為GraalVM
這篇文章主要介紹了用Spring Native將SpringBoot程序轉(zhuǎn)換為GraalVM的方法,幫助大家更好的理解和學(xué)習(xí)使用SpringBoot,感興趣的朋友可以了解下2021-04-04
Java 高并發(fā)九:鎖的優(yōu)化和注意事項(xiàng)詳解
本文主要介紹Java高并發(fā)鎖的優(yōu)化和注意事項(xiàng),這里整理了詳細(xì)的資料,并講解了 1. 鎖優(yōu)化的思路和方法 2. 虛擬機(jī)內(nèi)的鎖優(yōu)化 3. 一個(gè)錯(cuò)誤使用鎖的案例 4. ThreadLocal及其源碼分析等知識(shí),有需要的小伙伴可以參考下2016-09-09
Java Validation方法入?yún)⑿r?yàn)實(shí)現(xiàn)過程解析
這篇文章主要介紹了Java Validation方法入?yún)⑿r?yàn)實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11

