Java注解--实现简单读取excel
目录
实现工具类
基本使用
实现工具类
利用注解实现简单的 excel 数据读取,利用注解对类的属性和 excel 中的表头映射,使用 Apache 的 poi 就不用在业务代码中涉及 row,rows 这些属性了。
定义注解:
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface Excel {
String <span class="hljs-title function_">name</span><span class="hljs-params">()</span>;
}
由于本例中只涉及根据 Excel 表头部分对 Excel 进行解析,只定义了一个 name 作为和 Excel 表头的隐射。
工具类完整代码如下:
public class ExcelUtil<T> {
Class<T> clazz;
<span class="hljs-keyword">public</span> <span class="hljs-title function_">ExcelUtil</span><span class="hljs-params">(Class<T> clazz)</span> {
<span class="hljs-built_in">this</span>.clazz = clazz;
}
<span class="hljs-keyword">public</span> List<T> <span class="hljs-title function_">importExcel</span><span class="hljs-params">(String sheetName, InputStream input)</span> {
<span class="hljs-type">int</span> <span class="hljs-variable">maxCol</span> <span class="hljs-operator">=</span> <span class="hljs-number">0</span>;
List<T> list = <span class="hljs-keyword">new</span> <span class="hljs-title class_">ArrayList</span><T>();
<span class="hljs-keyword">try</span> {
<span class="hljs-type">Workbook</span> <span class="hljs-variable">workbook</span> <span class="hljs-operator">=</span> WorkbookFactory.create(input);
<span class="hljs-type">Sheet</span> <span class="hljs-variable">sheet</span> <span class="hljs-operator">=</span> workbook.getSheet(sheetName);
<span class="hljs-comment">// 如果指定sheet名,则取指定sheet中的内容.</span>
<span class="hljs-keyword">if</span> (!sheetName.trim().equals(<span class="hljs-string">""</span>)) {
sheet = workbook.getSheet(sheetName);
}
<span class="hljs-comment">// 如果传入的sheet名不存在则默认指向第1个sheet.</span>
<span class="hljs-keyword">if</span> (sheet == <span class="hljs-literal">null</span>) {
sheet = workbook.getSheetAt(<span class="hljs-number">0</span>);
}
<span class="hljs-type">int</span> <span class="hljs-variable">rows</span> <span class="hljs-operator">=</span> sheet.getPhysicalNumberOfRows();
<span class="hljs-comment">// 有数据时才处理</span>
<span class="hljs-keyword">if</span> (rows > <span class="hljs-number">0</span>) {
List<Field> allFields = getMappedFiled(clazz, <span class="hljs-literal">null</span>);
<span class="hljs-comment">// 定义一个map用于存放列的序号和field.</span>
Map<Integer, Field> fieldsMap = <span class="hljs-keyword">new</span> <span class="hljs-title class_">HashMap</span><Integer, Field>();
<span class="hljs-comment">// 第一行为表头</span>
<span class="hljs-type">Row</span> <span class="hljs-variable">rowHead</span> <span class="hljs-operator">=</span> sheet.getRow(<span class="hljs-number">0</span>);
Map<String, Integer> cellMap = <span class="hljs-keyword">new</span> <span class="hljs-title class_">HashMap</span><>();
<span class="hljs-type">int</span> <span class="hljs-variable">cellNum</span> <span class="hljs-operator">=</span> rowHead.getPhysicalNumberOfCells();
<span class="hljs-keyword">for</span> (<span class="hljs-type">int</span> <span class="hljs-variable">i</span> <span class="hljs-operator">=</span> <span class="hljs-number">0</span>; i < cellNum; i++){
cellMap.put(rowHead.getCell(i).getStringCellValue().toLowerCase(), i);
}
<span class="hljs-keyword">for</span> (Field field : allFields) {
<span class="hljs-comment">// 将有注解的field存放到map中.</span>
<span class="hljs-keyword">if</span> (field.isAnnotationPresent(Excel.class)) {
<span class="hljs-type">Excel</span> <span class="hljs-variable">attr</span> <span class="hljs-operator">=</span> field.getAnnotation(Excel.class);
<span class="hljs-comment">// 根据Name来获取相应的failed</span>
<span class="hljs-type">int</span> <span class="hljs-variable">col</span> <span class="hljs-operator">=</span> cellMap.get(attr.name().toLowerCase());
field.setAccessible(<span class="hljs-literal">true</span>);
fieldsMap.put(col, field);
}
}
<span class="hljs-comment">// 从第2行开始取数据</span>
<span class="hljs-keyword">for</span> (<span class="hljs-type">int</span> <span class="hljs-variable">i</span> <span class="hljs-operator">=</span> <span class="hljs-number">1</span>; i < rows; i++) {
<span class="hljs-type">Row</span> <span class="hljs-variable">row</span> <span class="hljs-operator">=</span> sheet.getRow(i);
<span class="hljs-type">T</span> <span class="hljs-variable">entity</span> <span class="hljs-operator">=</span> <span class="hljs-literal">null</span>;
<span class="hljs-keyword">for</span> (<span class="hljs-type">int</span> <span class="hljs-variable">j</span> <span class="hljs-operator">=</span> <span class="hljs-number">0</span>; j < cellNum; j++) {
<span class="hljs-type">Cell</span> <span class="hljs-variable">cell</span> <span class="hljs-operator">=</span> row.getCell(j);
<span class="hljs-keyword">if</span> (cell == <span class="hljs-literal">null</span>) {
<span class="hljs-keyword">continue</span>;
}
<span class="hljs-type">int</span> <span class="hljs-variable">cellType</span> <span class="hljs-operator">=</span> cell.getCellType();
<span class="hljs-type">String</span> <span class="hljs-variable">c</span> <span class="hljs-operator">=</span> <span class="hljs-string">""</span>;
<span class="hljs-keyword">if</span> (cellType == HSSFCell.CELL_TYPE_NUMERIC) {
<span class="hljs-type">DecimalFormat</span> <span class="hljs-variable">df</span> <span class="hljs-operator">=</span> <span class="hljs-keyword">new</span> <span class="hljs-title class_">DecimalFormat</span>(<span class="hljs-string">"0"</span>);
c = df.format(cell.getNumericCellValue());
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> (cellType == HSSFCell.CELL_TYPE_BOOLEAN) {
c = String.valueOf(cell.getBooleanCellValue());
} <span class="hljs-keyword">else</span> {
c = cell.getStringCellValue();
}
<span class="hljs-keyword">if</span> (c == <span class="hljs-literal">null</span> || c.equals(<span class="hljs-string">""</span>)) {
<span class="hljs-keyword">continue</span>;
}
entity = (entity == <span class="hljs-literal">null</span> ? clazz.newInstance() : entity);
<span class="hljs-comment">// 从map中得到对应列的field.</span>
<span class="hljs-type">Field</span> <span class="hljs-variable">field</span> <span class="hljs-operator">=</span> fieldsMap.get(j);
<span class="hljs-keyword">if</span> (field == <span class="hljs-literal">null</span>) {
<span class="hljs-keyword">continue</span>;
}
<span class="hljs-comment">// 取得类型,并根据对象类型设置值.</span>
Class<?> fieldType = field.getType();
<span class="hljs-keyword">if</span> (String.class == fieldType) {
field.set(entity, String.valueOf(c));
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ((Integer.TYPE == fieldType)
|| (Integer.class == fieldType)) {
field.set(entity, Integer.valueOf(c));
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ((Long.TYPE == fieldType)
|| (Long.class == fieldType)) {
field.set(entity, Long.valueOf(c));
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ((Float.TYPE == fieldType)
|| (Float.class == fieldType)) {
field.set(entity, Float.valueOf(c));
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ((Short.TYPE == fieldType)
|| (Short.class == fieldType)) {
field.set(entity, Short.valueOf(c));
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> ((Double.TYPE == fieldType)
|| (Double.class == fieldType)) {
field.set(entity, Double.valueOf(c));
} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> (Character.TYPE == fieldType) {
<span class="hljs-keyword">if</span> (c.length() > <span class="hljs-number">0</span>) {
field.set(entity, c.charAt(<span class="hljs-number">0</span>));
}
}
}
<span class="hljs-keyword">if</span> (entity != <span class="hljs-literal">null</span>) {
list.add(entity);
}
}
}
} <span class="hljs-keyword">catch</span> (Exception e) {
e.printStackTrace();
}
<span class="hljs-keyword">return</span> list;
}
<span class="hljs-comment">/**
* 得到实体类所有通过注解映射了数据表的字段
*
* <span class="hljs-doctag">@param</span> clazz
* <span class="hljs-doctag">@param</span> fields
* <span class="hljs-doctag">@return</span>
*/</span>
<span class="hljs-keyword">private</span> List<Field> <span class="hljs-title function_">getMappedFiled</span><span class="hljs-params">(Class clazz, List<Field> fields)</span> {
<span class="hljs-keyword">if</span> (fields == <span class="hljs-literal">null</span>) {
fields = <span class="hljs-keyword">new</span> <span class="hljs-title class_">ArrayList</span><Field>();
}
<span class="hljs-comment">// 得到所有定义字段</span>
Field[] allFields = clazz.getDeclaredFields();
<span class="hljs-comment">// 得到所有field并存放到一个list中.</span>
<span class="hljs-keyword">for</span> (Field field : allFields) {
<span class="hljs-keyword">if</span> (field.isAnnotationPresent(Excel.class)) {
fields.add(field);
}
}
<span class="hljs-keyword">if</span> (clazz.getSuperclass() != <span class="hljs-literal">null</span>
&& !clazz.getSuperclass().equals(Object.class)) {
getMappedFiled(clazz.getSuperclass(), fields);
}
<span class="hljs-keyword">return</span> fields;
}
}
代码很简单,获取 sheet,解析第一行,并和实体类标有注解的字段一一对应,用 hashMap 记录下来,然后循环取得 Excel 中剩下所有的数据,根据 map 的对应关系将值 set 到对应字段。
基本使用
待解析表格如下:
定义实体类:
public class User {
<span class="hljs-meta">@Excel(name = "filed1")</span>
<span class="hljs-keyword">private</span> String name;
<span class="hljs-meta">@Excel(name = "filed2")</span>
<span class="hljs-keyword">private</span> String nameEn;
<span class="hljs-meta">@Excel(name = "filed3")</span>
<span class="hljs-keyword">private</span> Integer age;
<span class="hljs-meta">@Excel(name = "filed4")</span>
<span class="hljs-keyword">private</span> String six;
<span class="hljs-meta">@Excel(name = "filed5")</span>
<span class="hljs-keyword">private</span> String weight;
<span class="hljs-comment">// ...getter setter</span>
}
使用工具类:
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
}
利用这个思路可以扩展出导出 excel 功能,利用注解指定导出的 excel 表头,甚至可以轻松控制 excel 表头的颜色,合并属性等等,在 xdemo 中有详细复杂的示例,可以研究下。由于我的需求很简单,就不整那么复杂啦。