当前位置: 首页 > news >正文

Workbook导出excel封装的工具类

  

  在实际中导出excel非常常见,于是自己封装了一个导出数据到excel的工具类,先附上代码,最后会写出实例和解释。支持03和07两个版本的 excel。

  HSSF导出的是xls的excel,XSSF导出的是xlsx的excel,SXSSF导出的也是xlsx的excel,只不过这个用于处理数据量大的情况,生成文件之后数据不会留在内存中。

  代码中依赖了slf4j日志包,commons-io包的IOUtils关闭流,commons-lang和commons-collections包等包。

package cn.xm.exam.utils;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.commons.collections.MapUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang.ArrayUtils;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFCellStyle;
import org.apache.poi.hssf.usermodel.HSSFFont;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.Font;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.streaming.SXSSFSheet;
import org.apache.poi.xssf.streaming.SXSSFWorkbook;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class ExcelExporter {

    private static final Logger LOGGER = LoggerFactory.getLogger(ExcelExporter.class);

    private String[] headerNames;
    private Workbook workBook;
    private Sheet sheet;

    /**
     * 
     * @param headerNames
     *            表头
     * @param sheetName
     *            sheet的名称
     * @param excelVerson
     *            excel的版本
     */
    public ExcelExporter(String[] headerNames, String sheetName, String excelVerson) {
        this.headerNames = headerNames;
        // 创建一个工作簿
        if ("07".equals(excelVerson)) {
            // workBook = new XSSFWorkbook();//处理07版本excel
            workBook = new SXSSFWorkbook();// 处理07版本,但是适用于大数据量,导出之后数据不会占用内存
        } else {
            workBook = new HSSFWorkbook();
        }
        // 创建一个工作表sheet
        sheet = workBook.createSheet(sheetName);
        initHeader();
    }

    /**
     * 初始化表头信息
     */
    private void initHeader() {
        // 创建第一行
        Row row = sheet.createRow(0);
        Cell cell = null;
        // 创建表头
        for (int i = 0; i < headerNames.length; i++) {
            cell = row.createCell(i);
            cell.setCellValue(headerNames[i]);
            setCellStyle(cell);
        }
    }

    /**
     * 设置单元格样式
     * 
     * @param cell
     *            单元格
     */
    public void setCellStyle(Cell cell) {
        // 设置样式
        CellStyle cellStyle = workBook.createCellStyle();
        cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER); // 设置字体居中
        // 设置字体
        Font font = workBook.createFont();
        font.setFontName("宋体");
        font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);// 字体加粗
        font.setFontHeightInPoints((short) 13);

        cellStyle.setFont(font);
        cell.setCellStyle(cellStyle);
    }

    /**
     * 创建行内容(每一行的数据装在list中)
     * 
     * @param datas
     *            每一行的数据
     * @param rowIndex
     *            行号(从1开始)
     */
    public void createTableRow(List<String> datas, int rowIndex) {
        // 创建第i行
        Row row = sheet.createRow(rowIndex);
        Cell cell = null;
        // 写入数据
        for (int index = 0, length = datas.size(); index < length; index++) {
            // 参数代表第几列
            cell = row.createCell(index);
            cell.setCellType(HSSFCell.CELL_TYPE_STRING);
            cell.setCellValue(datas.get(index));
        }
    }

    /**
     * 
     * @param datas
     *            数据,每一个map都是一行
     * @param keys
     *            key[i]代表从map中获取keys[i]的值作为第i列的值,如果传的是null默认取表头
     */
    public void createTableRows(List<Map<String, Object>> datas, String[] keys) {
        for (int i = 0, length_1 = datas.size(); i < length_1; i++) {
            if (ArrayUtils.isEmpty(keys)) {
                keys = headerNames;
            }
            // 创建行(从第二行开始)
            Map<String, Object> data = datas.get(i);
            Row row = sheet.createRow(i + 1);
            Cell cell = null;
            for (int j = 0, length_2 = keys.length; j < length_2; j++) {
                // 单元格获取map中的key
                String key = keys[j];
                String value = MapUtils.getString(data, key, "");

                cell = row.createCell(j);
                cell.setCellType(HSSFCell.CELL_TYPE_STRING);
                cell.setCellValue(value);
            }

        }
    }

    /**
     * 根据表头自动调整列宽度
     */
    public void autoAllSizeColumn() {
        if (sheet instanceof SXSSFSheet) {// 如果是SXSSFSheet,需要调用trackAllColumnsForAutoSizing方法一次
            SXSSFSheet tmpSheet = (SXSSFSheet) sheet;
            tmpSheet.trackAllColumnsForAutoSizing();
        }
        for (int i = 0, length = headerNames.length; i < length; i++) {
            sheet.autoSizeColumn(i);
        }
    }

    /**
     * 将数据写出到excel中
     * 
     * @param outputStream
     */
    public void exportExcel(OutputStream outputStream) {
        // 导出之前先自动设置列宽
        this.autoAllSizeColumn();
        try {
            workBook.write(outputStream);
        } catch (IOException e) {
            LOGGER.error(" exportExcel error", e);
        } finally {
            IOUtils.closeQuietly(outputStream);
        }
    }

}

  上面的代码逻辑非常简单,创建实例的时候就初始化表头信息和创建sheet。

  向excel中填充数据的方式有两种,就是上面的createTableRow方法和createTableRows方法。

  createTableRow(List,int)方法就是多次调用此方法,list中数据,int是行号。list中数据依次作为第i行的列数据。

  createTableRows(List<Map>,String[])这个方法应该是非常常用的一种方式。我们从数据库查询到的数据大多数映射为Map放入list中,因此上面的方法就比较常用。List<Map>参数就是所有的数据,一个Map代表一行,String[]是Map中的key,也就是数组的第一个元素对应的key作为第一列,第二个元素是作为第二列。如果map数据的key正好与表头一致我们可以传一个null。(因为Map是基于数组+链表,且存入的是无序的,所以无法直接通过map中的key确定列。除非传的数据是LinkedHashMap)。这个方法封装的比较好,如果一条数据中没有对应的值会将此单元格设为空。

 

下面是自己测试代码:

(1)测试List<String>写入数据,导出03版本的excel

    public static void test1() {
        ExcelExporter hssfWorkExcel = new ExcelExporter(new String[] { "姓名", "年龄" }, "人员基本信息", "03");
        for (int i = 0; i < 10; i++) {
            List<String> data = new ArrayList<>();
            data.add("namesssssssssssssss水水水水水水水水水水水水水水水水水水水ssssssssssssssss" + i);
            data.add("" + (i + 20));
            hssfWorkExcel.createTableRow(data, i + 1);
        }
        try {
            hssfWorkExcel.exportExcel(new FileOutputStream(new File("e:/test.xls")));
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
    }

结果:

 

 (2)测试List<Map>写入数据,导出07版本的excel

    public static void test2() {
        HSSFWorkExcel hssfWorkExcel = new HSSFWorkExcel(new String[] { "姓名", "年龄" }, "人员基本信息","07");
        List<Map<String, Object>> datas = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
            Map data = new HashMap<>();
            data.put("name", "tttttttttttttt" + i);
            data.put("age", "age" + i);
            datas.add(data);
        }
        hssfWorkExcel.createTableRows(datas, new String[] { "name", "age" });
        try {
            hssfWorkExcel.exportExcel(new FileOutputStream(new File("e:/test1.xls")));
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
    }

 结果:

 

下面附上一段自己实际中用到的代码:

果然在导出excel的时候方便多了,只需要短短的几行代码就可以生成一个excel到本地。

package cn.xm.exam.action.safeHat;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.text.ParseException;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.collections.MapUtils;
import org.apache.commons.io.FileUtils;
import org.apache.commons.lang.StringUtils;
import org.apache.commons.lang3.time.DateFormatUtils;
import org.apache.commons.lang3.time.DateUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;

import com.opensymphony.xwork2.ActionSupport;

import cn.xm.exam.service.safehat.SafehatService;
import cn.xm.exam.utils.ExamSystemUtils;
import cn.xm.exam.utils.HSSFWorkExcel;
import cn.xm.exam.utils.ValidateCheck;

/**
 * 导出安全帽台账
 * 
 * @author Administrator
 *
 */
@Controller
public class ExtSafeHatTaizhang extends ActionSupport {
    private static final Logger log = LoggerFactory.getLogger(ExtSafeHatTaizhang.class);
    private String userName;
    private String idCard;
    private String safeHatNum;
    private String fileName;

    @Autowired
    private SafehatService safehatService;

    @Override
    public String execute() throws Exception {
        return SUCCESS;
    }

    public InputStream getInputStream() {
        // 获取一个临时文件
        File file = ExamSystemUtils.getTmpFile();

        // 只用返回一个输入流
        try {
            // 查数据
            List<Map<String, Object>> safehatTaizhang = getSafehatTaizhang();

            // 写入文件中
            String[] headerNames = new String[] { "序号", "年龄", "性别", "学历", "身体状况", "职务(工种)", "成绩", "身份证号", "安全帽编号",
                    "备注" };
            HSSFWorkExcel hssfWorkExcel = new HSSFWorkExcel(headerNames, "安全帽台账信息","03");
            String[] keys = new String[] { "index", "age", "sex", "empeducate", "empphysicalstatus", "empType",
                    "thirdScore", "idCode", "safeHatNum", "desc" };
            hssfWorkExcel.createTableRows(safehatTaizhang, keys);
            hssfWorkExcel.exportExcel(new FileOutputStream(file));

            // 返回流
            return FileUtils.openInputStream(file);

        } catch (IOException e) {
            log.error("getInputStream error", e);
        }
        return null;
    }

    public List<Map<String, Object>> getSafehatTaizhang() {
        Map condition = new HashMap<>();
        if (ValidateCheck.isNotNull(userName)) {
            condition.put("userName", userName);
        }
        if (ValidateCheck.isNotNull(idCard)) {
            condition.put("idCard", idCard);
        }
        if (ValidateCheck.isNotNull(safeHatNum)) {
            condition.put("safeHatNum", safeHatNum);
        }
        // 开始分页
        List<Map<String, Object>> results = null;
        try {
            results = safehatService.getSafehatTaizhang(condition);
        } catch (Exception e) {
            log.error("getSafehatTaizhang error", e);
            results = new ArrayList<>();
            return results;
        }
        if (CollectionUtils.isNotEmpty(results)) {
            disPoseResults(results);
        }

        return results;
    }

    private void disPoseResults(List<Map<String, Object>> results) {
        int index = 1;
        for (Map map : results) {
            // 装一个序号进去
            map.put("index", index++);

            // 处理性别
            if (StringUtils.isNotBlank(MapUtils.getString(map, "sex"))) {
                if ("1".equals(map.get("sex").toString())) {
                    map.put("sex", "男");
                }
                if ("2".equals(map.get("sex").toString())) {
                    map.put("sex", "女");
                }
            }

            // 处理年龄
            String birthday = MapUtils.getString(map, "birthday");
            if (StringUtils.isNotBlank(birthday)) {
                Date date = null;
                try {
                    date = DateUtils.parseDateStrictly(birthday, "yyyy-MM-dd");
                } catch (ParseException e) {
                    date = new Date();
                }
                int birhYear = date.getYear();
                int nowYear = (new Date()).getYear();
                map.put("age", (nowYear - birhYear) + 1);
            }
        }
    }

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public String getIdCard() {
        return idCard;
    }

    public void setIdCard(String idCard) {
        this.idCard = idCard;
    }

    public String getSafeHatNum() {
        return safeHatNum;
    }

    public void setSafeHatNum(String safeHatNum) {
        this.safeHatNum = safeHatNum;
    }

    public SafehatService getSafehatService() {
        return safehatService;
    }

    public void setSafehatService(SafehatService safehatService) {
        this.safehatService = safehatService;
    }

    public String getFileName() {
        String name = "安全帽台账" + DateFormatUtils.format(new Date(), "yyyy-MM-dd") + ".xls";
        try {
            name = new String(name.getBytes("UTF-8"), "ISO8859-1");
        } catch (UnsupportedEncodingException e) {
            log.error("getFileName error", e);
            return DateFormatUtils.format(new Date(), "yyyy-MM-dd") + ".xls";
        }
        return name;
    }
}

 

相关文章:

  • 勒索病毒防范方案-有韩立刚老师总结的非常规手段
  • js导出excel文件
  • Java枚举比较用equals还是==
  • vue两个组件间值的传递或修改方式
  • 如何在国内构建一个硅谷级的高效技术团队?
  • C#中的结构体与类的区别 (转载)
  • 如何用纯 CSS 创作一个记事本翻页动画
  • SpringBoot整合Angular应用第三弹-渲染RestAPI数据
  • CentOS从零开始部署Nodejs项目
  • d6
  • Jeff Bean谈Flink与流式处理的5大新发现
  • 解决 scapy “NameError: global name 'wrpcap' is not defined” 错误
  • 【xshell】xshell设置快捷键 设置Ctrl+C Ctrl+V快捷键为复制粘贴
  • Android FlatBuffers数据交互
  • 【man】 查看命令帮助文档
  • 【Redis学习笔记】2018-06-28 redis命令源码学习1
  • 【干货分享】SpringCloud微服务架构分布式组件如何共享session对象
  • 07.Android之多媒体问题
  • 2018以太坊智能合约编程语言solidity的最佳IDEs
  • create-react-app做的留言板
  • CSS居中完全指南——构建CSS居中决策树
  • PHP那些事儿
  • vue-loader 源码解析系列之 selector
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • ------- 计算机网络基础
  • 前端 CSS : 5# 纯 CSS 实现24小时超市
  • 如何选择开源的机器学习框架?
  • 系统认识JavaScript正则表达式
  • 与 ConTeXt MkIV 官方文档的接驳
  • 远离DoS攻击 Windows Server 2016发布DNS政策
  • 怎么把视频里的音乐提取出来
  • Python 之网络式编程
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • 如何通过报表单元格右键控制报表跳转到不同链接地址 ...
  • 如何在招聘中考核.NET架构师
  • ​用户画像从0到100的构建思路
  • #Lua:Lua调用C++生成的DLL库
  • #Z2294. 打印树的直径
  • #在 README.md 中生成项目目录结构
  • (04)Hive的相关概念——order by 、sort by、distribute by 、cluster by
  • (11)MATLAB PCA+SVM 人脸识别
  • (BFS)hdoj2377-Bus Pass
  • (附源码)spring boot建达集团公司平台 毕业设计 141538
  • (力扣题库)跳跃游戏II(c++)
  • (一)VirtualBox安装增强功能
  • (已解决)什么是vue导航守卫
  • (转)Android学习笔记 --- android任务栈和启动模式
  • (转)fock函数详解
  • (转)socket Aio demo
  • (轉貼) 資訊相關科系畢業的學生,未來會是什麼樣子?(Misc)
  • *** 2003
  • ./include/caffe/util/cudnn.hpp: In function ‘const char* cudnnGetErrorString(cudnnStatus_t)’: ./incl
  • .NET 3.0 Framework已经被添加到WindowUpdate
  • .NET HttpWebRequest、WebClient、HttpClient
  • .Net IOC框架入门之一 Unity