Jeesite导出Excel源码解析

转载自https://www.aliyun.com/jiaocheng/1340693.html

导入导出的公共工具

/** 
 * Copyright © 2012-2013 JeeSite All rights reserved. 
 * Licensed under the Apache License, Version 2.0 (the "License"); 
 */  
package com.cyou.seal.common.utils.excel.annotation;  
  
import java.lang.annotation.ElementType;  
import java.lang.annotation.Retention;  
import java.lang.annotation.RetentionPolicy;  
import java.lang.annotation.Target;  
  
/** 
 * Excel注解定义 
 * @author ThinkGem 
 * @version 2013-03-10 
 */  
  
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.TYPE})  
@Retention(RetentionPolicy.RUNTIME)  
public @interface ExcelField {  
  
    /** 
     * 导出字段名(默认调用当前字段的“get”方法,如指定导出字段为对象,请填写“对象名.对象属性”,例:“area.name”、“office.name”) 
     */  
    String value() default "";  
      
    /** 
     * 导出字段标题(需要添加批注请用“**”分隔,标题**批注,仅对导出模板有效) 
     */  
    String title();  
      
    /** 
     * 字段类型(0:导出导入;1:仅导出;2:仅导入) 
     */  
    int type() default 0;  
  
    /** 
     * 导出字段对齐方式(0:自动;1:靠左;2:居中;3:靠右) 
     *  
     * 备注:Integer/Long类型设置居右对齐(align=3) 
     */  
    int align() default 0;  
      
    /** 
     * 导出字段字段排序(升序) 
     */  
    int sort() default 0;  
  
    /** 
     * 如果是字典类型,请设置字典的type值 
     */  
    String dictType() default "";  
      
    /** 
     * 反射类型 
     */  
    Class fieldType() default Class.class;  
      
    /** 
     * 字段归属组(根据分组导出导入) 
     */  
    int[] groups() default {};  
}  

反射工具类:

/** 
 * 反射工具类. 
 * 提供调用getter/setter方法, 访问私有变量, 调用私有方法, 获取泛型类型Class, 被AOP过的真实类等工具函数. 
 * @author calvin 
 * @version 2013-01-15 
 */  
@SuppressWarnings("rawtypes")  
public class Reflections {  
      
    private static final String SETTER_PREFIX = "set";  
  
    private static final String GETTER_PREFIX = "get";  
  
    private static final String CGLIB_CLASS_SEPARATOR = "$$";  
      
    private static Logger logger = LoggerFactory.getLogger(Reflections.class);  
  
    /** 
     * 调用Getter方法. 
     * 支持多级,如:对象名.对象名.方法 
     */  
    public static Object invokeGetter(Object obj, String propertyName) {  
        Object object = obj;  
        for (String name : StringUtils.split(propertyName, ".")){  
            String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(name);  
            object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {});  
        }  
        return object;  
    }  
  
    /** 
     * 调用Setter方法, 仅匹配方法名。 
     * 支持多级,如:对象名.对象名.方法 
     */  
    public static void invokeSetter(Object obj, String propertyName, Object value) {  
        Object object = obj;  
        String[] names = StringUtils.split(propertyName, ".");  
        for (int i=0; i             if(i                 String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(names[i]);  
                object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {});  
            }else{  
                String setterMethodName = SETTER_PREFIX + StringUtils.capitalize(names[i]);  
                invokeMethodByName(object, setterMethodName, new Object[] { value });  
            }  
        }  
    }  
  
    /** 
     * 直接读取对象属性值, 无视private/protected修饰符, 不经过getter函数. 
     */  
    public static Object getFieldValue(final Object obj, final String fieldName) {  
        Field field = getAccessibleField(obj, fieldName);  
  
        if (field == null) {  
            throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + obj + "]");  
        }  
  
        Object result = null;  
        try {  
            result = field.get(obj);  
        } catch (IllegalAccessException e) {  
            logger.error("不可能抛出的异常{}", e.getMessage());  
        }  
        return result;  
    }  
  
    /** 
     * 直接设置对象属性值, 无视private/protected修饰符, 不经过setter函数. 
     */  
    public static void setFieldValue(final Object obj, final String fieldName, final Object value) {  
        Field field = getAccessibleField(obj, fieldName);  
  
        if (field == null) {  
            throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + obj + "]");  
        }  
  
        try {  
            field.set(obj, value);  
        } catch (IllegalAccessException e) {  
            logger.error("不可能抛出的异常:{}", e.getMessage());  
        }  
    }  
  
    /** 
     * 直接调用对象方法, 无视private/protected修饰符. 
     * 用于一次性调用的情况,否则应使用getAccessibleMethod()函数获得Method后反复调用. 
     * 同时匹配方法名+参数类型, 
     */  
    public static Object invokeMethod(final Object obj, final String methodName, final Class[] parameterTypes,  
            final Object[] args) {  
        Method method = getAccessibleMethod(obj, methodName, parameterTypes);  
        if (method == null) {  
            throw new IllegalArgumentException("Could not find method [" + methodName + "] on target [" + obj + "]");  
        }  
  
        try {  
            return method.invoke(obj, args);  
        } catch (Exception e) {  
            throw convertReflectionExceptionToUnchecked(e);  
        }  
    }  
  
    /** 
     * 直接调用对象方法, 无视private/protected修饰符, 
     * 用于一次性调用的情况,否则应使用getAccessibleMethodByName()函数获得Method后反复调用. 
     * 只匹配函数名,如果有多个同名函数调用第一个。 
     */  
    public static Object invokeMethodByName(final Object obj, final String methodName, final Object[] args) {  
        Method method = getAccessibleMethodByName(obj, methodName);  
        if (method == null) {  
            throw new IllegalArgumentException("Could not find method [" + methodName + "] on target [" + obj + "]");  
        }  
  
        try {  
            return method.invoke(obj, args);  
        } catch (Exception e) {  
            throw convertReflectionExceptionToUnchecked(e);  
        }  
    }  
  
    /** 
     * 循环向上转型, 获取对象的DeclaredField, 并强制设置为可访问. 
     *  
     * 如向上转型到Object仍无法找到, 返回null. 
     */  
    public static Field getAccessibleField(final Object obj, final String fieldName) {  
        Validate.notNull(obj, "object can't be null");  
        Validate.notBlank(fieldName, "fieldName can't be blank");  
        for (Class superClass = obj.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()) {  
            try {  
                Field field = superClass.getDeclaredField(fieldName);  
                makeAccessible(field);  
                return field;  
            } catch (NoSuchFieldException e) {//NOSONAR  
                // Field不在当前类定义,继续向上转型  
                continue;// new add  
            }  
        }  
        return null;  
    }  
  
    /** 
     * 循环向上转型, 获取对象的DeclaredMethod,并强制设置为可访问. 
     * 如向上转型到Object仍无法找到, 返回null. 
     * 匹配函数名+参数类型。 
     *  
     * 用于方法需要被多次调用的情况. 先使用本函数先取得Method,然后调用Method.invoke(Object obj, Object... args) 
     */  
    public static Method getAccessibleMethod(final Object obj, final String methodName,  
            final Class... parameterTypes) {  
        Validate.notNull(obj, "object can't be null");  
        Validate.notBlank(methodName, "methodName can't be blank");  
  
        for (Class searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) {  
            try {  
                Method method = searchType.getDeclaredMethod(methodName, parameterTypes);  
                makeAccessible(method);  
                return method;  
            } catch (NoSuchMethodException e) {  
                // Method不在当前类定义,继续向上转型  
                continue;// new add  
            }  
        }  
        return null;  
    }  
  
    /** 
     * 循环向上转型, 获取对象的DeclaredMethod,并强制设置为可访问. 
     * 如向上转型到Object仍无法找到, 返回null. 
     * 只匹配函数名。 
     *  
     * 用于方法需要被多次调用的情况. 先使用本函数先取得Method,然后调用Method.invoke(Object obj, Object... args) 
     */  
    public static Method getAccessibleMethodByName(final Object obj, final String methodName) {  
        Validate.notNull(obj, "object can't be null");  
        Validate.notBlank(methodName, "methodName can't be blank");  
  
        for (Class searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) {  
            Method[] methods = searchType.getDeclaredMethods();  
            for (Method method : methods) {  
                if (method.getName().equals(methodName)) {  
                    makeAccessible(method);  
                    return method;  
                }  
            }  
        }  
        return null;  
    }  
  
    /** 
     * 改变private/protected的方法为public,尽量不调用实际改动的语句,避免JDK的SecurityManager抱怨。 
     */  
    public static void makeAccessible(Method method) {  
        if ((!Modifier.isPublic(method.getModifiers()) || !Modifier.isPublic(method.getDeclaringClass().getModifiers()))  
                && !method.isAccessible()) {  
            method.setAccessible(true);  
        }  
    }  
  
    /** 
     * 改变private/protected的成员变量为public,尽量不调用实际改动的语句,避免JDK的SecurityManager抱怨。 
     */  
    public static void makeAccessible(Field field) {  
        if ((!Modifier.isPublic(field.getModifiers()) || !Modifier.isPublic(field.getDeclaringClass().getModifiers()) || Modifier  
                .isFinal(field.getModifiers())) && !field.isAccessible()) {  
            field.setAccessible(true);  
        }  
    }  
  
    /** 
     * 通过反射, 获得Class定义中声明的泛型参数的类型, 注意泛型必须定义在父类处 
     * 如无法找到, 返回Object.class. 
     * eg. 
     * public UserDao extends HibernateDao 
     * 
     * @param clazz The class to introspect 
     * @return the first generic declaration, or Object.class if cannot be determined 
     */  
    @SuppressWarnings("unchecked")  
    public static Class getClassGenricType(final Class clazz) {  
        return getClassGenricType(clazz, 0);  
    }  
  
    /** 
     * 通过反射, 获得Class定义中声明的父类的泛型参数的类型. 
     * 如无法找到, 返回Object.class. 
     *  
     * 如public UserDao extends HibernateDao 
     * 
     * @param clazz clazz The class to introspect 
     * @param index the Index of the generic ddeclaration,start from 0. 
     * @return the index generic declaration, or Object.class if cannot be determined 
     */  
    public static Class getClassGenricType(final Class clazz, final int index) {  
  
        Type genType = clazz.getGenericSuperclass();  
  
        if (!(genType instanceof ParameterizedType)) {  
            logger.warn(clazz.getSimpleName() + "'s superclass not ParameterizedType");  
            return Object.class;  
        }  
  
        Type[] params = ((ParameterizedType) genType).getActualTypeArguments();  
  
        if (index >= params.length || index < 0) {  
            logger.warn("Index: " + index + ", Size of " + clazz.getSimpleName() + "'s Parameterized Type: "  
                    + params.length);  
            return Object.class;  
        }  
        if (!(params[index] instanceof Class)) {  
            logger.warn(clazz.getSimpleName() + " not set the actual class on superclass generic parameter");  
            return Object.class;  
        }  
  
        return (Class) params[index];  
    }  
      
    public static Class getUserClass(Object instance) {  
        Assert.notNull(instance, "Instance must not be null");  
        Class clazz = instance.getClass();  
        if (clazz != null && clazz.getName().contains(CGLIB_CLASS_SEPARATOR)) {  
            Class superClass = clazz.getSuperclass();  
            if (superClass != null && !Object.class.equals(superClass)) {  
                return superClass;  
            }  
        }  
        return clazz;  
  
    }  
      
    /** 
     * 将反射时的checked exception转换为unchecked exception. 
     */  
    public static RuntimeException convertReflectionExceptionToUnchecked(Exception e) {  
        if (e instanceof IllegalAccessException || e instanceof IllegalArgumentException  
                || e instanceof NoSuchMethodException) {  
            return new IllegalArgumentException(e);  
        } else if (e instanceof InvocationTargetException) {  
            return new RuntimeException(((InvocationTargetException) e).getTargetException());  
        } else if (e instanceof RuntimeException) {  
            return (RuntimeException) e;  
        }  
        return new RuntimeException("Unexpected Checked Exception.", e);  
    }  
}  

关于自定义注解:

(1)@Target:说明了Annotation所修饰的对象范围,Annotation可以用于packages、types(类、接口、枚举、Annotation类型),类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了target可更加明晰其修饰的目标。取值(ElementType)有:

1.CONSTRUCTOR:用于描述构造器

    2.FIELD:用于描述域

    3.LOCAL_VARIABLE:用于描述局部变量

    4.METHOD:用于描述方法

    5.PACKAGE:用于描述包

    6.PARAMETER:用于描述参数

    7.TYPE:用于描述类、接口(包括注解类型) 或enum声明    

(2)@Retention定义了该Annotation被保留的时间长短:
某些Annotation仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;
编译在class文件中的Annotation可能会被虚拟机忽略,而另一些在class被装载时将被读取(请注意并不影响class的执行,因为Annotation与class在使用上是被分离的)。
使用这个meta-Annotation可以对 Annotation的“生命周期”限制。
取值(RetentionPoicy)有:
    1.SOURCE:在源文件中有效(即源文件保留)
    2.CLASS:在class文件中有效(即class保留)
    3.RUNTIME:在运行时有效(即运行时保留)

这个工具类,可以帮注解可以帮我我们定义导出的时候的字段名,标题,对齐方式,字段排序,反射的是哪个类,是导入还是导出。


Excel导出
我们在Excel导出的时候,基本思路就是,(1)获取要导出的数据   (2)通过反射获取对应的class的字段  (3)将导出的数据通过反射放到list (4)绘制Excel

导出工具类:
/** 
 * 导出Excel文件(导出“XLSX”格式,支持大数据量导出   @see org.apache.poi.ss.SpreadsheetVersion) 
 * @author ThinkGem 
 * @version 2013-04-21 
 */  
public class ExportExcel {  
      
    private static Logger log = LoggerFactory.getLogger(ExportExcel.class);  
              
    /** 
     * 工作薄对象 
     */  
    private SXSSFWorkbook wb;  
      
    /** 
     * 工作表对象 
     */  
    private Sheet sheet;  
      
    /** 
     * 样式列表 
     */  
    private Map styles;  
      
    /** 
     * 当前行号 
     */  
    private int rownum;  
      
    /** 
     * 注解列表(Object[]{ ExcelField, Field/Method }) 
     */  
    List annotationList = Lists.newArrayList();  
      
    /** 
     * 构造函数 
     * @param title 表格标题,传“空值”,表示无标题 
     * @param cls 实体对象,通过annotation.ExportField获取标题 
     */  
    public ExportExcel(String title, Class cls){  
        this(title, cls, 2);  
    }  
      
    /** 
     * 构造函数 
     * @param title 表格标题,传“空值”,表示无标题 
     * @param cls 实体对象,通过annotation.ExportField获取标题 
     * @param type 导出类型(1:导出数据;2:导出模板) 
     * @param groups 导入分组 
     */  
    public ExportExcel(String title, Class cls, int type, int... groups){  
        // Get annotation field   
        //getDeclaredFields:通过反射获取对应class的全部字段,包括私有字段,但是不包括父类申明的字段  
        Field[] fs = cls.getDeclaredFields();  
        for (Field f : fs){  
            //获取字段中带有@ExcelField的标签  
            ExcelField ef = f.getAnnotation(ExcelField.class);  
            //标签不为空并且导出类型是导入导出或者导出模板  
            if (ef != null && (ef.type()==0 || ef.type()==type)){  
                if (groups!=null && groups.length>0){  
                    boolean inGroup = false;  
                    for (int g : groups){  
                        if (inGroup){  
                            break;  
                        }  
                        for (int efg : ef.groups()){  
                            if (g == efg){  
                                inGroup = true;  
                                annotationList.add(new Object[]{ef, f});  
                                break;  
                            }  
                        }  
                    }  
                }else{  
                    annotationList.add(new Object[]{ef, f});  
                }  
            }  
        }  
        // Get annotation method  
        //获取对应类中的全部方法,包括pulbic,protected,private.但是不包括继承的方法,当然也包括他所实现接口的方法  
        Method[] ms = cls.getDeclaredMethods();  
        for (Method m : ms){  
            ExcelField ef = m.getAnnotation(ExcelField.class);  
            //字段:导入导出或者字段导出  
            if (ef != null && (ef.type()==0 || ef.type()==type)){  
                if (groups!=null && groups.length>0){  
                    boolean inGroup = false;  
                    for (int g : groups){  
                        if (inGroup){  
                            break;  
                        }  
                        for (int efg : ef.groups()){  
                            if (g == efg){  
                                inGroup = true;  
                                annotationList.add(new Object[]{ef, m});  
                                break;  
                            }  
                        }  
                    }  
                }else{  
                    annotationList.add(new Object[]{ef, m});  
                }  
            }  
        }  
        // Field sorting   
        Collections.sort(annotationList, new Comparator() {  
            public int compare(Object[] o1, Object[] o2) {  
                return new Integer(((ExcelField)o1[0]).sort()).compareTo(  
                        new Integer(((ExcelField)o2[0]).sort()));  
            };  
        });  
        // Initialize  
        List headerList = Lists.newArrayList();  
        for (Object[] os : annotationList){  
            String t = ((ExcelField)os[0]).title();  
            // 如果是导出,则去掉注释  
            if (type==1){  
                String[] ss = StringUtils.split(t, "**", 2);  
                if (ss.length==2){  
                    t = ss[0];  
                }  
            }  
            headerList.add(t);  
        }  
        initialize(title, headerList);  
    }  
      
    /** 
     * 构造函数 
     * @param title 表格标题,传“空值”,表示无标题 
     * @param headers 表头数组 
     */  
    public ExportExcel(String title, String[] headers) {  
        initialize(title, Lists.newArrayList(headers));  
    }  
      
    /** 
     * 构造函数 
     * @param title 表格标题,传“空值”,表示无标题 
     * @param headerList 表头列表 
     */  
    public ExportExcel(String title, List headerList) {  
        initialize(title, headerList);  
    }  
      
    /** 
     * 初始化函数 
     * @param title 表格标题,传“空值”,表示无标题 
     * @param headerList 表头列表 
     */  
    private void initialize(String title, List headerList) {  
        //SXSSFWorkbook专门用来处理大数据写入Excel2007的问题  
        this.wb = new SXSSFWorkbook(500);  
        this.sheet = wb.createSheet("Export");  
        this.styles = createStyles(wb);  
        // Create title  
        if (StringUtils.isNotBlank(title)){  
            Row titleRow = sheet.createRow(rownum++);  
            titleRow.setHeightInPoints(30);//设置行高  
            Cell titleCell = titleRow.createCell(0);  
            titleCell.setCellStyle(styles.get("title"));//将已经写好的单元格属性调用  
            titleCell.setCellValue(title);//设置单元格的值  
            //addMergedRegion:设置列宽,设置列宽的方法在HSSFSheet中,方法参数:1:第几列,2:宽度  
            //单元格合并方法也是在HSSFSheet中,方法参数:一个CellRangeAddress,该类构造函数的4个参数分别表示为:合并开始行,合并结束行,合并开始列,合并结束列  
            sheet.addMergedRegion(new CellRangeAddress(titleRow.getRowNum(),  
                    titleRow.getRowNum(), titleRow.getRowNum(), headerList.size()-1));  
        }  
        // Create header  
        if (headerList == null){  
            throw new RuntimeException("headerList not null!");  
        }  
        Row headerRow = sheet.createRow(rownum++);  
        headerRow.setHeightInPoints(16);  
        for (int i = 0; i < headerList.size(); i++) {  
            Cell cell = headerRow.createCell(i);  
            cell.setCellStyle(styles.get("header"));  
            String[] ss = StringUtils.split(headerList.get(i), "**", 2);  
            if (ss.length==2){  
                cell.setCellValue(ss[0]);  
                Comment comment = this.sheet.createDrawingPatriarch().createCellComment(  
                        new XSSFClientAnchor(0, 0, 0, 0, (short) 3, 3, (short) 5, 6));  
                comment.setString(new XSSFRichTextString(ss[1]));  
                cell.setCellComment(comment);  
            }else{  
                cell.setCellValue(headerList.get(i));  
            }  
            sheet.autoSizeColumn(i);  
        }  
        for (int i = 0; i < headerList.size(); i++) {    
            int colWidth = sheet.getColumnWidth(i)*2;  
            sheet.setColumnWidth(i, colWidth < 3000 ? 3000 : colWidth);    
        }  
        log.debug("Initialize success.");  
    }  
      
    /** 
     * 创建表格样式 
     * @param wb 工作薄对象 
     * @return 样式列表 
     */  
    private Map createStyles(Workbook wb) {  
        Map styles = new HashMap();  
          
        CellStyle style = wb.createCellStyle();  
        style.setAlignment(CellStyle.ALIGN_CENTER);  
        style.setVerticalAlignment(CellStyle.VERTICAL_CENTER);  
        Font titleFont = wb.createFont();  
        titleFont.setFontName("Arial");  
        titleFont.setFontHeightInPoints((short) 16);  
        titleFont.setBoldweight(Font.BOLDWEIGHT_BOLD);  
        style.setFont(titleFont);  
        styles.put("title", style);  
  
        style = wb.createCellStyle();  
        style.setVerticalAlignment(CellStyle.VERTICAL_CENTER);  
        style.setBorderRight(CellStyle.BORDER_THIN);  
        style.setRightBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());  
        style.setBorderLeft(CellStyle.BORDER_THIN);  
        style.setLeftBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());  
        style.setBorderTop(CellStyle.BORDER_THIN);  
        style.setTopBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());  
        style.setBorderBottom(CellStyle.BORDER_THIN);  
        style.setBottomBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());  
        Font dataFont = wb.createFont();  
        dataFont.setFontName("Arial");  
        dataFont.setFontHeightInPoints((short) 10);  
        style.setFont(dataFont);  
        styles.put("data", style);  
          
        style = wb.createCellStyle();  
        style.cloneStyleFrom(styles.get("data"));  
        style.setAlignment(CellStyle.ALIGN_LEFT);  
        styles.put("data1", style);  
  
        style = wb.createCellStyle();  
        style.cloneStyleFrom(styles.get("data"));  
        style.setAlignment(CellStyle.ALIGN_CENTER);  
        styles.put("data2", style);  
  
        style = wb.createCellStyle();  
        style.cloneStyleFrom(styles.get("data"));  
        style.setAlignment(CellStyle.ALIGN_RIGHT);  
        styles.put("data3", style);  
          
        style = wb.createCellStyle();  
        style.cloneStyleFrom(styles.get("data"));  
//      style.setWrapText(true);  
        style.setAlignment(CellStyle.ALIGN_CENTER);  
        style.setFillForegroundColor(IndexedColors.GREY_50_PERCENT.getIndex());  
        style.setFillPattern(CellStyle.SOLID_FOREGROUND);  
        Font headerFont = wb.createFont();  
        headerFont.setFontName("Arial");  
        headerFont.setFontHeightInPoints((short) 10);  
        headerFont.setBoldweight(Font.BOLDWEIGHT_BOLD);  
        headerFont.setColor(IndexedColors.WHITE.getIndex());  
        style.setFont(headerFont);  
        styles.put("header", style);  
          
        return styles;  
    }  
  
    /** 
     * 添加一行 
     * @return 行对象 
     */  
    public Row addRow(){  
        return sheet.createRow(rownum++);  
    }  
      
  
    /** 
     * 添加一个单元格 
     * @param row 添加的行 
     * @param column 添加列号 
     * @param val 添加值 
     * @return 单元格对象 
     */  
    public Cell addCell(Row row, int column, Object val){  
        return this.addCell(row, column, val, 0, Class.class);  
    }  
      
    /** 
     * 添加一个单元格 
     * @param row 添加的行 
     * @param column 添加列号 
     * @param val 添加值 
     * @param align 对齐方式(1:靠左;2:居中;3:靠右) 
     * @return 单元格对象 
     */  
    public Cell addCell(Row row, int column, Object val, int align, Class fieldType){  
        Cell cell = row.createCell(column);  
        CellStyle style = styles.get("data"+(align>=1&&align<=3?align:""));  
        try {  
            if (val == null){  
                cell.setCellValue("");  
            } else if (val instanceof String) {  
                cell.setCellValue((String) val);  
            } else if (val instanceof Integer) {  
                cell.setCellValue((Integer) val);  
            } else if (val instanceof Long) {  
                cell.setCellValue((Long) val);  
            } else if (val instanceof Double) {  
                cell.setCellValue((Double) val);  
            } else if (val instanceof Float) {  
                cell.setCellValue((Float) val);  
            } else if (val instanceof Date) {  
                DataFormat format = wb.createDataFormat();  
                style.setDataFormat(format.getFormat("yyyy-MM-dd"));  
                cell.setCellValue((Date) val);  
            } else {  
                if (fieldType != Class.class){  
                    cell.setCellValue((String)fieldType.getMethod("setValue", Object.class).invoke(null, val));  
                }else{  
                    cell.setCellValue((String)Class.forName(this.getClass().getName().replaceAll(this.getClass().getSimpleName(),   
                        "fieldtype."+val.getClass().getSimpleName()+"Type")).getMethod("setValue", Object.class).invoke(null, val));  
                }  
            }  
        } catch (Exception ex) {  
            log.info("Set cell value ["+row.getRowNum()+","+column+"] error: " + ex.toString());  
            cell.setCellValue(val.toString());  
        }  
        cell.setCellStyle(style);  
        return cell;  
    }  
  
    /** 
     * 添加数据(通过annotation.ExportField添加数据) 
     * @return list 数据列表 
     */  
    public ExportExcel setDataList(List list){  
        for (E e : list){  
            int colunm = 0;  
            Row row = this.addRow();  
            StringBuilder sb = new StringBuilder();  
            for (Object[] os : annotationList){  
                ExcelField ef = (ExcelField)os[0];  
                Object val = null;  
                // Get entity value  
                try{  
                    if (StringUtils.isNotBlank(ef.value())){  
                        val = Reflections.invokeGetter(e, ef.value());  
                    }else{  
                        if (os[1] instanceof Field){  
                            val = Reflections.invokeGetter(e, ((Field)os[1]).getName());  
                        }else if (os[1] instanceof Method){  
                            val = Reflections.invokeMethod(e, ((Method)os[1]).getName(), new Class[] {}, new Object[] {});  
                        }  
                    }  
                    // If is dict, get dict label  
                    if (StringUtils.isNotBlank(ef.dictType())){  
                        val = DictUtils.getDictLabel(val==null?"":val.toString(), ef.dictType(), "");  
                    }  
                }catch(Exception ex) {  
                    // Failure to ignore  
                    log.info(ex.toString());  
                    val = "";  
                }  
                this.addCell(row, colunm++, val, ef.align(), ef.fieldType());  
                sb.append(val + ", ");  
            }  
            //log.debug("Write success: ["+row.getRowNum()+"] "+sb.toString());  
        }  
        return this;  
    }  
      
    public ExportExcel setDataListList(List> listList){  
        for (List list : listList) {  
            int colunm = 0;  
            Row row = this.addRow();  
            StringBuilder sb = new StringBuilder();  
            for (String val : list) {  
                this.addCell(row, colunm++, val, 2, "".getClass());  
                sb.append(val + ", ");  
            }  
        }  
        return this;  
    }  
    /** 
     * 输出数据流 
     * @param os 输出数据流 
     */  
    public ExportExcel write(OutputStream os) throws IOException{  
        wb.write(os);  
        return this;  
    }  
      
    /** 
     * 输出到客户端 
     * @param fileName 输出文件名 
     */  
    public ExportExcel write(HttpServletResponse response, String fileName) throws IOException{  
        response.reset();  
        response.setContentType("application/octet-stream; charset=utf-8");  
        response.setHeader("Content-Disposition", "attachment; filename="+Encodes.urlEncode(fileName));  
        write(response.getOutputStream());  
        return this;  
    }  
      
    /** 
     * 输出到文件 
     * @param fileName 输出文件名 
     */  
    public ExportExcel writeFile(String name) throws FileNotFoundException, IOException{  
        FileOutputStream os = new FileOutputStream(name);  
        this.write(os);  
        return this;  
    }  
      
    /** 
     * 清理临时文件 
     */  
    public ExportExcel dispose(){  
        wb.dispose();  
        return this;  
    }  
      
//  /**  
//   * 导出测试  
//   */  
    public static void main(String[] args) throws Throwable {  
        String[] titles = { "商品名", "商品单价", "商品单位" };   
        List headerList = Lists.newArrayList();  
        for (int i = 0; i <= titles.length-1; i++) {  
            headerList.add(titles[i]);  
        }  
          
        List dataRowList = Lists.newArrayList();  
        for (int i = 1; i <= headerList.size(); i++) {  
            dataRowList.add("数据"+i);  
        }  
          
        List> dataList = Lists.newArrayList();  
        for (int i = 1; i <=100; i++) {  
            dataList.add(dataRowList);  
        }  
  
        ExportExcel ee = new ExportExcel("表格标题", headerList);  
          
        for (int i = 0; i < dataList.size(); i++) {  
            Row row = ee.addRow();  
            for (int j = 0; j < dataList.get(i).size(); j++) {  
                ee.addCell(row, j, dataList.get(i).get(j));  
            }  
        }  
          
        ee.writeFile("C:/target/export.xlsx");  
  
        ee.dispose();  
          
        log.debug("Export success.");  
          
    }  
  
}  

导出实体类Vo:将要导出的实体类的字段的get方法上加上我们的自定义注解,申明导出的title,导出顺序,是否字典,对齐方式:


public class Certificate {  
      
    private static final long serialVersionUID = -2627501790384505697L;  
      
    /** 
     * 资产唯一编码 
     */  
    @LogFiledName(value="资产唯一编码")  
    private String assertUniqueCode;  
      
    /** 
     * 版本类型 
     */  
    @LogFiledName(value="版本类型")  
    private String versionType;  
      
    /** 
     * 期限(字典:永久/租赁) 
     */  
    @LogFiledName(value="期限(字典:永久/租赁)",dic="certificate_term")  
    private String term;  
      
    public Certificate() {  
        super();  
    }  
  
    public Certificate(String id){  
        this();  
        this.id = id;  
    }  
      
    @ExcelField(title="资产唯一编码", align=2, sort=1)  
    public String getAssertUniqueCode() {  
        return assertUniqueCode;  
    }  
  
    public void setAssertUniqueCode(String assertUniqueCode) {  
        this.assertUniqueCode = assertUniqueCode;  
    }  
  
    @ExcelField(title="版本类型", align=2, sort=4)  
    public String getVersionType() {  
        return versionType;  
    }  
  
    public void setVersionType(String versionType) {  
        this.versionType = versionType;  
    }  
  
    @ExcelField(title="期限", align=2, sort=5, dictType="certificate_term")  
    public String getTerm() {  
        return term;  
    }  
  
    public void setTerm(String term) {  
        this.term = term;  
    }  
  
  
}  

对上面的代码简单分析:
(1)当我们获取到全部的数据,调用如下:
new ExportExcel("软资产信息", Certificate.class).setDataList(certificateListAll).write(response, fileName).dispose();  
调用ExportExcel的构造函数,参数1:标题,参数2:实体对象。

(2)通过反射获取class的全部字段,包括私有字段,但是不包括父类申明的字段:
Field[] fs = cls.getDeclaredFields();

(3)遍历全部字段,取出带有@ExcelFiel标签的字段,这样我们就可以得到都导出那些字段了:
ExcelField ef = f.getAnnotation(ExcelField.class);
  
(4)遍历我们取出的标签

(5)遍历类中全部方法,包括public,protected,private。但是不包括继承的方法,当然也包括他所实现接口的方法:
Method[] ms = cls.getDeclaredMethods();  
同样,遍历所以带有注解的方法

(6)将我们遍历筛选出来的List进行排序

(7)调用初始化函数,绘制Sheet
private void initialize(String title, List headerList)  
先创建SXSSFWorkbook,在创建Sheet,将Sheet使用统一的样式:this.styles = createStyles(wb);
设置标题是第几行,行高,创建单元格,合并单元格。

(8)添加数据:public ExportExcel setDataList(List list)
遍历我们筛选出的数据,设置第几行,如果是字段,返回字典对应的label。添加单元格:
public Cell addCell(Row row, int column, Object val, int align, Class fieldType)  
创建单元格:Cell cell = row.createCell(column);
判断导出的值的类型,并设置到单元格中.

(9)最后以文件的方式输出到客户端:设置文件的格式,文件名
public ExportExcel write(HttpServletResponse response, String fileName)

 

 

注意:当type是numberic的时候,又分为数字格式和日期格式。
数字格式->如果是double类型就要读取double类型,如果是int则直接读取。使用getNumericCellValue读取数字类型。
日期格式->则需要使用POI提供的DataFormatter类的formatCellValue方法来读取日期类型单元格。以下工具类中,出现date类型和数字类型冲突的问题,导致数字类型显示为日期型

具体解决方案:在excelExport.java中做一下修改

/** 
     * 添加一个单元格 
     * @param row 添加的行 
     * @param column 添加列号 
     * @param val 添加值 
     * @param align 对齐方式(1:靠左;2:居中;3:靠右) 
     * @return 单元格对象 
     */  
    public Cell addCell(Row row, int column, Object val, int align, Class fieldType){

        CellStyle dateStyle = wb.createCellStyle();

        dateStyle.cloneStyleFrom(style);

        Cell cell = row.createCell(column);  
        CellStyle style = styles.get("data"+(align>=1&&align<=3?align:""));  

        boolean isNotDateCell = true;
        try {  
            if (val == null){  
                cell.setCellValue("");  
            } else if (val instanceof String) {  
                cell.setCellValue((String) val);  
            } else if (val instanceof Integer) {  
                cell.setCellValue((Integer) val);  
            } else if (val instanceof Long) {  
                cell.setCellValue((Long) val);  
            } else if (val instanceof Double) {  
                cell.setCellValue((Double) val);  
            } else if (val instanceof Float) {  
                cell.setCellValue((Float) val);  
            } else if (val instanceof Date) {  
                DataFormat format = wb.createDataFormat();                  
                dateStyle.setDataFormat(format.getFormat("yyyy-MM-dd"));  
                cell.setCellValue((Date) val);

                cell.setCellStyle(dateStyle);

                 isNotDateCell = false; 
            } else {  
                if (fieldType != Class.class){  
                    cell.setCellValue((String)fieldType.getMethod("setValue", Object.class).invoke(null, val));  
                }else{  
                    cell.setCellValue((String)Class.forName(this.getClass().getName().replaceAll(this.getClass().getSimpleName(),   
                        "fieldtype."+val.getClass().getSimpleName()+"Type")).getMethod("setValue", Object.class).invoke(null, val));  
                }  
            }  
        } catch (Exception ex) {  
            log.info("Set cell value ["+row.getRowNum()+","+column+"] error: " + ex.toString());  
            cell.setCellValue(val.toString());  
        }  

        if (isNotDateCell) {
               cell.setCellStyle(style);
        }

        cell.setCellStyle(style);  
        return cell;  
    }  

你可能感兴趣的:(spring)