POI实现excel的导入导出-[采用反射实现的版本]

前段时间学习了POI实现excel的导入导出-[未采用反射实现的版本]的文章,觉得不采用反射实现非常不合理,在祥哥【我眼中的大佬】的指导下实现了POI实现excel的导入导出-[采用反射实现的版本]的实现。
A、具体的要点如下:


1、采用泛型,增加代码的灵活性和可扩展性;
2、使用适配器模式;
3、使用反射自动执行对应字段的方法;
4、绑定字段的值和表头文字【采用linkedhashmap()】实现;

B、代码


ExcelVO.java

package com.spring.boot.gp4zj.util;

import java.util.List;
import java.util.Map;

/**
 * excel导入导出相关的必须信息;
 * 
 * @author xdsm
 *
 */
public class ExcelVO<T> {
    // 表头列表;
    Map<String, String> workBookHeaderFields = null;
    // excel中的数据值对象;
    List<T> workBookVos = null;
    // 泛型数据对象;
    T clazzRes = null;

    public List<T> getWorkBookVos() {
        return workBookVos;
    }

    public void setWorkBookVos(List<T> workBookVos) {
        this.workBookVos = workBookVos;
    }

    public Map<String, String> getWorkBookHeaderFields() {
        return workBookHeaderFields;
    }

    public void setWorkBookHeaderFields(Map<String, String> workBookHeaderFields) {
        this.workBookHeaderFields = workBookHeaderFields;
    }

    public T getClazzRes() {
        return clazzRes;
    }

    public void setClazzRes(T clazzRes) {
        this.clazzRes = clazzRes;
    }
}

POIUtils.java

package com.spring.boot.gp4zj.util;

import java.io.FileInputStream;
import java.io.InputStream;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFCellStyle;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;

import com.spring.boot.gp4zj.model.AttenceDetail;
import com.spring.boot.gp4zj.model.ExcelVOMapper;

/**
 * Excel操作相关的工具类;
 * 
 * @author:JunZhou
 * @Company:LongZheng
 * @Email:1769676159@qq.com
 * @2018年1月16日@下午3:19:25
 */
public class POIUtils2 {
    public static void main(String[] args) throws Exception {
        /*
         * AttenceDetail clazzRes = new AttenceDetail(); // 测试数据;
         * ExcelVO<AttenceDetail> excelVo = new ExcelVO<AttenceDetail>();
         * excelVo.setClazzRes(clazzRes);
         * 
         * @SuppressWarnings("unused") List<String> workBookHeaderFields = new
         * ArrayList<String>(); final Map<String, String> HEADER_FIELDS = new
         * LinkedHashMap<String, String>(); HEADER_FIELDS.put("考勤日期",
         * "attencedTime"); HEADER_FIELDS.put("考勤名称", "attencerName");
         * HEADER_FIELDS.put("员工编码", "attencerCode"); HEADER_FIELDS.put("出勤时长",
         * "workTimeLen"); HEADER_FIELDS.put("加班时长", "extraTimeLen");
         * HEADER_FIELDS.put("合计出勤时长", "totalTimeLen");
         * HEADER_FIELDS.put("报工单位", "applyCompany"); HEADER_FIELDS.put("结算单位",
         * "payCompany"); HEADER_FIELDS.put("创建人", "creator");
         * HEADER_FIELDS.put("出勤人天", "workDayNum"); HEADER_FIELDS.put("加班人天",
         * "extraDayNum"); HEADER_FIELDS.put("合计人天", "totalDayNum");
         * HEADER_FIELDS.put("是否删除汇总表", "isDelGather");
         * HEADER_FIELDS.put("是否确认", "isConfirmed"); HEADER_FIELDS.put("创建时间",
         * "createTime"); HEADER_FIELDS.put("修改时间", "editTime"); // 生成测试数据;
         * excelVo.setWorkBookHeaderFields(HEADER_FIELDS); List<AttenceDetail>
         * workBookVos = TestDataGenerater.attenceDetailTestDatasGenerater(2);
         * excelVo.setWorkBookVos(workBookVos); createExcel(excelVo);
         */
        String excelPath = "C:/Users/xdsm/Desktop/Excel-900500219.xls";
        AttenceDetail attenceDetailClazz = new AttenceDetail();
        List<AttenceDetail> attenctDetailList = readExcel(excelPath, attenceDetailClazz);
        for (AttenceDetail attenceDetail : attenctDetailList) {
            System.out.println(attenceDetail);
        }
    }

    /**
     * 创建Excel
     * 
     * @param <T>
     * 
     * @param list
     *            数据
     */
    @SuppressWarnings({ "unchecked", "unused" })
    public static <T> HSSFWorkbook createExcel(ExcelVO<T> excelVo) {
        @SuppressWarnings("rawtypes")
        Class clazz = excelVo.getClazzRes().getClass();
        // 表头信息列表;
        List<String> workBookHeaderFields = new ArrayList<String>();
        Map<String, String> headerFields = excelVo.getWorkBookHeaderFields();
        // 表格数据信息列表;
        List<T> dataList = excelVo.getWorkBookVos();
        // 创建一个Excel文件
        HSSFWorkbook workbook = new HSSFWorkbook();
        // 创建一个工作表
        HSSFSheet sheet = workbook.createSheet("考勤明细表");
        // 添加表头行
        HSSFRow hssfRow = sheet.createRow(0);
        // 设置单元格格式居中
        HSSFCellStyle cellStyle = workbook.createCellStyle();
        cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER);
        // 添加表头内容
        // -------第一行第0-2列---------
        // 设置表头的相关信息;
        int headTitleIndex = 0;
        HSSFCell headCell = null;// 此处使用同一个引用对象,可以节省内存;
        for (Entry<String, String> vo : headerFields.entrySet()) {
            // 表头的名称字段;
            if (headTitleIndex == 0) {
                headCell = hssfRow.createCell(headTitleIndex);
                headCell.setCellValue(vo.getKey());
                headCell.setCellStyle(cellStyle);
            } else {
                headCell = hssfRow.createCell(headTitleIndex);
                headCell.setCellValue(vo.getKey());
                headCell.setCellStyle(cellStyle);
            }
            headTitleIndex++;
        }
        // 填充数据到工作簿;
        for (int i = 0; i < dataList.size(); i++) {
            // 创建数据行;
            hssfRow = sheet.createRow((int) i + 1);
            T attenceDetail = dataList.get(i);
            // 创建单元格,并设置值
            int fieldVoIndex = 0;
            HSSFCell cell = null;// 此处使用同一个引用对象,可以节省内存;
            for (Entry<String, String> vo : headerFields.entrySet()) {
                // 字段值取值器;
                String fieldVoGetter = "get" + (vo.getValue()).substring(0, 1).toUpperCase()
                        + (vo.getValue()).substring(1);
                cell = hssfRow.createCell(fieldVoIndex);
                // TODO Auto-generated
                Method method;
                try {
                    String invokeResultStr = null;
                    method = clazz.getMethod(fieldVoGetter);
                    Object methodResult = method.invoke(attenceDetail);
                    if (methodResult != null) {
                        invokeResultStr = methodResult.toString();
                    }
                    cell.setCellValue(invokeResultStr);
                    cell.setCellStyle(cellStyle);
                } catch (Exception e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
                // method.invoke(clazz.newInstance());
                fieldVoIndex++;
            }
        }

        return workbook;
    }

    /**
     * 接收上传的excel并读取其中的信息;
     * 
     * @throws IllegalAccessException
     * @throws Exception
     */
    @SuppressWarnings({ "unchecked", "rawtypes" })
    public static <T> List<T> readExcel(String excelPath, T clazzRes) throws Exception {
        Class clazz = clazzRes.getClass();
        List<T> list = new ArrayList<T>();
        HSSFWorkbook workbook = null;
        try {
            // 读取Excel文件
            InputStream inputStream = new FileInputStream(excelPath);
            workbook = new HSSFWorkbook(inputStream);
            inputStream.close();
        } catch (Exception e) {
            e.printStackTrace();
        }

        // 循环工作表
        for (int numSheet = 0; numSheet < workbook.getNumberOfSheets(); numSheet++) {
            HSSFSheet hssfSheet = workbook.getSheetAt(numSheet);
            if (hssfSheet == null) {
                continue;
            }
            // 循环行
            for (int rowNum = 1; rowNum < hssfSheet.getLastRowNum(); rowNum++) {
                HSSFRow hssfRow = hssfSheet.getRow(rowNum);
                if (hssfRow == null) {
                    continue;
                }
                // 填充数据到行;
                Map<String, String> HEADER_FIELDS = ExcelVOMapper.getAttenceDetailMapper();
                HSSFCell cell = null;// 公用cell,节省内存的开销;
                // 每行的数据对象;
                T dataVo = (T) clazz.newInstance();
                int fieldVoIndex = 0;
                for (Entry<String, String> vo : HEADER_FIELDS.entrySet()) {
                    // 字段值取值器;
                    String fieldVoSetter = "set" + (vo.getValue()).substring(0, 1).toUpperCase()
                            + (vo.getValue()).substring(1);
                    // TODO Auto-generated
                    Method method;
                    try {
                        cell = hssfRow.getCell(fieldVoIndex);
                        if (cell == null) {
                            continue;
                        }
                        method = clazz.getMethod(fieldVoSetter, String.class);
                        String cellValue = cell.getStringCellValue();
                        /*
                         * if(cellValue==null||"".equals(cellValue)){ // 空数据列测试;
                         * if (fieldVoIndex == 1) { throw new
                         * MyException("员工编码不能为空"); } }
                         */
                        // System.out.println(cellValue);
                        method.invoke(dataVo, cellValue);
                    } catch (Exception e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    fieldVoIndex++;
                }
                list.add(dataVo);
            }
        }
        return list;
    }

    /**
     * 获取指定数据类型的所有声明字段;
     * 
     * @param dataObj:泛型对象;
     * @return:返回字段列表【String】;
     */
    public static <T> List<String> getAllFields(T dataObj) {
        Class<? extends Object> dataObjClazz = dataObj.getClass();
        Field[] fields = dataObjClazz.getDeclaredFields();
        List<String> fieldsList = new ArrayList<String>();
        for (int i = 0; i < fields.length; i++) {
            String fieldStr = fields[i].toString();
            String realFieldStr = fieldStr.substring(fieldStr.lastIndexOf(".") + 1);
            fieldsList.add(realFieldStr);
        }
        return fieldsList;
    }

    /**
     * 获取对象中的所有已声明的方法名;
     * 
     * @param dataObj:泛型对象;
     * @return:返回方法列表【String】;
     */
    public static <T> List<String> getAllMethods(T dataObj) {
        List<String> methodsList = new ArrayList<String>();
        Class<? extends Object> dataObjClazz = dataObj.getClass();
        Method[] methods = dataObjClazz.getDeclaredMethods();
        for (int i = 0; i < methods.length; i++) {
            String fieldStr = methods[i].toString();
            if (fieldStr.contains("get")) {
                String realFieldStr = fieldStr.substring(fieldStr.lastIndexOf(".") + 1, fieldStr.lastIndexOf("("));
                methodsList.add(realFieldStr);
            }
        }
        return methodsList;
    }

    /**
     * 获取对象中的所有已声明的方法名;
     * 
     * @param dataObj:泛型对象;
     * @return:返回方法列表【String】;
     */
    public static <T> List<String> getAllSetMethods(T dataObj) {
        List<String> methodsList = new ArrayList<String>();
        @SuppressWarnings("unused")
        Class<? extends Object> dataObjClazz = dataObj.getClass();
        List<String> methods = getAllFields(dataObj);
        for (int i = 0; i < methods.size(); i++) {
            String fieldStr = methods.get(i).toString();
            String methodName = "set" + fieldStr.substring(0, 1).toUpperCase() + fieldStr.substring(1);
            methodsList.add(methodName);
        }
        return methodsList;
    }
}

C、使用要点


1、本工具类的使用前提是所有的字段均采用驼峰命名方式,且前两个字母为小写【因为eclipse自动生成的getter和setter会根据前两个字母进行自适应处理,并非固定模式】
2、本工具类并未对导入数据进行全面的非空或者类型等校验,具体的校验可自行添加。
3、本工具类因为采用的是反射是想,因此执行效率较低;
D、参考文章.
设计模式–观察者模式初探和java Observable模式
EasyPoi教程V1.0


Previous
POI实现excel的导入导出-[未采用反射实现的版本] POI实现excel的导入导出-[未采用反射实现的版本]
最近在做项目中遇到了使用POI导入导出excel的需求,所以在这里总结下,方便自己和朋友学习:1.经典博客: 1.JavaWEB–POI之EXCEL操作、优化、封装详解系列(一)–概述与原理 2.公共POI导出Excel方法–j
2018-12-04 Pursue
Next
POI的基本使用 POI的基本使用
POI的基本使用 首先,理解一下一个Excel的文件的组织形式,一个Excel文件对应于一个workbook(HSSFWorkbook),一个workbook可以有多个sheet(HSSFSheet)组成,一个sheet是由多个row(HS
2018-12-04 Pursue