java利用注解實(shí)現(xiàn)簡單的excel數(shù)據(jù)讀取
實(shí)現(xiàn)工具類
利用注解實(shí)現(xiàn)簡單的excel數(shù)據(jù)讀取,利用注解對類的屬性和excel中的表頭映射,使用Apache的poi就不用在業(yè)務(wù)代碼中涉及row,rows這些屬性了。
定義注解:
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface Excel {
String name();
}
由于本例中只涉及根據(jù)Excel表頭部分對Excel進(jìn)行解析,只定義了一個(gè)name作為和Excel表頭的隱射。
工具類完整代碼如下:
public class ExcelUtil<T> {
Class<T> clazz;
public ExcelUtil(Class<T> clazz) {
this.clazz = clazz;
}
public List<T> importExcel(String sheetName, InputStream input) {
int maxCol = 0;
List<T> list = new ArrayList<T>();
try {
Workbook workbook = WorkbookFactory.create(input);
Sheet sheet = workbook.getSheet(sheetName);
// 如果指定sheet名,則取指定sheet中的內(nèi)容.
if (!sheetName.trim().equals("")) {
sheet = workbook.getSheet(sheetName);
}
// 如果傳入的sheet名不存在則默認(rèn)指向第1個(gè)sheet.
if (sheet == null) {
sheet = workbook.getSheetAt(0);
}
int rows = sheet.getPhysicalNumberOfRows();
// 有數(shù)據(jù)時(shí)才處理
if (rows > 0) {
List<Field> allFields = getMappedFiled(clazz, null);
// 定義一個(gè)map用于存放列的序號和field.
Map<Integer, Field> fieldsMap = new HashMap<Integer, Field>();
// 第一行為表頭
Row rowHead = sheet.getRow(0);
Map<String, Integer> cellMap = new HashMap<>();
int cellNum = rowHead.getPhysicalNumberOfCells();
for (int i = 0; i < cellNum; i++){
cellMap.put(rowHead.getCell(i).getStringCellValue().toLowerCase(), i);
}
for (Field field : allFields) {
// 將有注解的field存放到map中.
if (field.isAnnotationPresent(Excel.class)) {
Excel attr = field.getAnnotation(Excel.class);
// 根據(jù)Name來獲取相應(yīng)的failed
int col = cellMap.get(attr.name().toLowerCase());
field.setAccessible(true);
fieldsMap.put(col, field);
}
}
// 從第2行開始取數(shù)據(jù)
for (int i = 1; i < rows; i++) {
Row row = sheet.getRow(i);
T entity = null;
for (int j = 0; j < cellNum; j++) {
Cell cell = row.getCell(j);
if (cell == null) {
continue;
}
int cellType = cell.getCellType();
String c = "";
if (cellType == HSSFCell.CELL_TYPE_NUMERIC) {
DecimalFormat df = new DecimalFormat("0");
c = df.format(cell.getNumericCellValue());
} else if (cellType == HSSFCell.CELL_TYPE_BOOLEAN) {
c = String.valueOf(cell.getBooleanCellValue());
} else {
c = cell.getStringCellValue();
}
if (c == null || c.equals("")) {
continue;
}
entity = (entity == null ? clazz.newInstance() : entity);
// 從map中得到對應(yīng)列的field.
Field field = fieldsMap.get(j);
if (field == null) {
continue;
}
// 取得類型,并根據(jù)對象類型設(shè)置值.
Class<?> fieldType = field.getType();
if (String.class == fieldType) {
field.set(entity, String.valueOf(c));
} else if ((Integer.TYPE == fieldType)
|| (Integer.class == fieldType)) {
field.set(entity, Integer.valueOf(c));
} else if ((Long.TYPE == fieldType)
|| (Long.class == fieldType)) {
field.set(entity, Long.valueOf(c));
} else if ((Float.TYPE == fieldType)
|| (Float.class == fieldType)) {
field.set(entity, Float.valueOf(c));
} else if ((Short.TYPE == fieldType)
|| (Short.class == fieldType)) {
field.set(entity, Short.valueOf(c));
} else if ((Double.TYPE == fieldType)
|| (Double.class == fieldType)) {
field.set(entity, Double.valueOf(c));
} else if (Character.TYPE == fieldType) {
if (c.length() > 0) {
field.set(entity, c.charAt(0));
}
}
}
if (entity != null) {
list.add(entity);
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
return list;
}
/**
* 得到實(shí)體類所有通過注解映射了數(shù)據(jù)表的字段
*
* @param clazz
* @param fields
* @return
*/
private List<Field> getMappedFiled(Class clazz, List<Field> fields) {
if (fields == null) {
fields = new ArrayList<Field>();
}
// 得到所有定義字段
Field[] allFields = clazz.getDeclaredFields();
// 得到所有field并存放到一個(gè)list中.
for (Field field : allFields) {
if (field.isAnnotationPresent(Excel.class)) {
fields.add(field);
}
}
if (clazz.getSuperclass() != null
&& !clazz.getSuperclass().equals(Object.class)) {
getMappedFiled(clazz.getSuperclass(), fields);
}
return fields;
}
}
代碼很簡單,獲取sheet,解析第一行,并和實(shí)體類標(biāo)有注解的字段一一對應(yīng),用hashMap記錄下來,然后循環(huán)取得Excel中剩下所有的數(shù)據(jù),根據(jù)map的對應(yīng)關(guān)系將值set到對應(yīng)字段。
基本使用
待解析表格如下:

定義實(shí)體類:
public class User {
@Excel(name = "filed1")
private String name;
@Excel(name = "filed2")
private String nameEn;
@Excel(name = "filed3")
private Integer age;
@Excel(name = "filed4")
private String six;
@Excel(name = "filed5")
private String weight;
// ...getter setter
}
使用工具類:
public static void main (String[] args) {
FileInputStream fileInputStream = null;
try {
fileInputStream = new FileInputStream("D://data.xlsx");
} catch (FileNotFoundException e) {
e.printStackTrace();
}
ExcelUtil<User> util = new ExcelUtil<>(User.class);
List<User> jalanHotelList = util.importExcel("user", fileInputStream);
// do something
}
利用這個(gè)思路可以擴(kuò)展出導(dǎo)出excel功能,利用注解指定導(dǎo)出的excel表頭,甚至可以輕松控制excel表頭的顏色,合并屬性等等,在xdemo中有詳細(xì)復(fù)雜的示例,可以研究下。由于我的需求很簡單,就不整那么復(fù)雜啦。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Spring Boot應(yīng)用監(jiān)控的實(shí)戰(zhàn)教程
Spring Boot 提供運(yùn)行時(shí)的應(yīng)用監(jiān)控和管理功能,下面這篇文章主要給大家介紹了關(guān)于Spring Boot應(yīng)用監(jiān)控的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-05-05
IDEA中的yml文件與properties互相轉(zhuǎn)換
這篇文章主要介紹了IDEA中的yml文件與properties互相轉(zhuǎn)換方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
關(guān)于Java實(shí)現(xiàn)HttpServer模擬前端接口調(diào)用
這篇文章主要介紹了關(guān)于Java實(shí)現(xiàn)Http?Server模擬前端接口調(diào)用,Http?協(xié)議是建立在?TCP?協(xié)議之上的協(xié)議,所以能用?TCP?來自己模擬一個(gè)簡單的?Http?Server?當(dāng)然是可以的,需要的朋友可以參考下2023-04-04
spring/springboot整合dubbo詳細(xì)教程
今天教大家如何使用spring/springboot整合dubbo,文中有非常詳細(xì)的圖文介紹及代碼示例,對正在學(xué)習(xí)java的小伙伴有很好地幫助,需要的朋友可以參考下2021-05-05
深入理解java內(nèi)置鎖(synchronized)和顯式鎖(ReentrantLock)
這篇文章主要介紹了Java多線程之內(nèi)置鎖(synchronized)和顯式鎖(ReentrantLock)的深入理解新的和用法,具有一定參考價(jià)值,需要的朋友可以了解下。2017-11-11
java日期格式化YYYY-MM-dd遇坑指南小結(jié)
本文主要介紹了java日期格式化YYYY-MM-dd遇坑指南小結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
詳解Java的TCP/IP編程學(xué)習(xí)--基于定界符的成幀
這篇文章主要介紹了Java的TCP/IP編程學(xué)習(xí)--基于定界符的成幀,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04

