Jeesite开发,Excel导入导出的功能

xiaoxiao2021-02-27  132

一. 导入导出的公共工具:

[java] view plain copy /**  * Copyright © 2012-2013 <a href="https://github.com/thinkgem/jeesite">JeeSite</a> 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 {};  }  

反射工具类:

[java] view plain copy /**  * 反射工具类.  * 提供调用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<names.length; i++){              if(i<names.length-1){                  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<User>      *      * @param clazz The class to introspect      * @return the first generic declaration, or Object.class if cannot be determined      */      @SuppressWarnings("unchecked")      public static <T> Class<T> getClassGenricType(final Class clazz) {          return getClassGenricType(clazz, 0);      }        /**      * 通过反射, 获得Class定义中声明的父类的泛型参数的类型.      * 如无法找到, 返回Object.class.      *       * 如public UserDao extends HibernateDao<User,Long>      *      * @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

导出工具类:

[java] view plain copy /**  * 导出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<String, CellStyle> styles;            /**      * 当前行号      */      private int rownum;            /**      * 注解列表(Object[]{ ExcelField, Field/Method })      */      List<Object[]> 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<Object[]>() {              public int compare(Object[] o1, Object[] o2) {                  return new Integer(((ExcelField)o1[0]).sort()).compareTo(                          new Integer(((ExcelField)o2[0]).sort()));              };          });          // Initialize          List<String> 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<String> headerList) {          initialize(title, headerList);      }            /**      * 初始化函数      * @param title 表格标题,传“空值”,表示无标题      * @param headerList 表头列表      */      private void initialize(String title, List<String> 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(0000, (short33, (short56));                  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<String, CellStyle> createStyles(Workbook wb) {          Map<String, CellStyle> styles = new HashMap<String, CellStyle>();                    CellStyle style = wb.createCellStyle();          style.setAlignment(CellStyle.ALIGN_CENTER);          style.setVerticalAlignment(CellStyle.VERTICAL_CENTER);          Font titleFont = wb.createFont();          titleFont.setFontName("Arial");          titleFont.setFontHeightInPoints((short16);          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((short10);          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((short10);          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 <E> ExportExcel setDataList(List<E> 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[1instanceof Field){                              val = Reflections.invokeGetter(e, ((Field)os[1]).getName());                          }else if (os[1instanceof 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<List<String>> listList){          for (List<String> 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<String> headerList = Lists.newArrayList();          for (int i = 0; i <= titles.length-1; i++) {              headerList.add(titles[i]);          }                    List<String> dataRowList = Lists.newArrayList();          for (int i = 1; i <= headerList.size(); i++) {              dataRowList.add("数据"+i);          }                    List<List<String>> 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,导出顺序,是否字典,对齐方式:

[java] view plain copy 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)当我们获取到全部的数据,调用如下:

[java] view plain copy new ExportExcel("软资产信息", Certificate.class).setDataList(certificateListAll).write(response, fileName).dispose();   调用ExportExcel的构造函数,参数1:标题,参数2:实体对象。 (2)通过反射获取class的全部字段,包括私有字段,但是不包括父类申明的字段: [java] view plain copy Field[] fs = cls.getDeclaredFields();   (3)遍历全部字段,取出带有@ExcelFiel标签的字段,这样我们就可以得到都导出那些字段了: [java] view plain copy ExcelField ef = f.getAnnotation(ExcelField.class);   (4)遍历我们取出的标签

(5)遍历类中全部方法,包括public,protected,private。但是不包括继承的方法,当然也包括他所实现接口的方法:

[java] view plain copy Method[] ms = cls.getDeclaredMethods();   同样,遍历所以带有注解的方法

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

(7)调用初始化函数,绘制Sheet

[java] view plain copy private void initialize(String title, List<String> headerList)   先创建SXSSFWorkbook,在创建Sheet,将Sheet使用统一的样式:this.styles = createStyles(wb);

设置标题是第几行,行高,创建单元格,合并单元格。

(8)添加数据:public <E> ExportExcel setDataList(List<E> list)

遍历我们筛选出的数据,设置第几行,如果是字段,返回字典对应的label。添加单元格:

[java] view plain copy 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)
转载请注明原文地址: https://www.6miu.com/read-15784.html

最新回复(0)