Java使用POI导出文件受损打不开

Java使用POI导出文件受损打不开

最近项目里涉及到了一个功能需要将数据库的表定时转换为Excel表格并发送给公司的对应负责人,作为开发者这种事情肯定要想一个一劳永逸的方法,所以这项功能的开发也就随之开展了

第一部分:Java使用POI生成Excel并保存本地

遇到问题:Java使用POI导出文件受损打不开

在本地测试和发送邮件全都没有问题,测试完成后上线到Linux服务器,中间遇到了路径变换的问题,简单的更改路径解决;上线后发送邮件成功,附件添加成功,邮箱中预览失败,下载到本地打开失败,错误提示文件受损。

解决方案:在网上找了很多解决方法,以下一一例举

1.IO流并不是每次都能读到1024个字节,所有用len作为每次读取数据的长度,否则会出现文件损坏的错误(不是这个)
2.设置扩展头,当Content-Type 的类型为要下载的类型时 , 这个信息头会告诉浏览器这个文件的名字和类型。HttpServletResponse 的res.setHeader(“Content-Disposition”, “attachment;filename=” + name);(没用到网络下载,也不是这个)
还有Linux目录没有读写权限等等~等等…

找了一圈发现网上没有处理方法,于是自己找原因解决,
首先考虑的是发送邮件中文件读取出现问题,但之前看了运行日志没有错误信息;
再看目录下POI生成的文件,使用连接工具打开失败,在Linux用vim命令可以打开,排除文件本身的损坏问题。
然后怀疑编码格式,因为文件名是根据表数据生成的,包含了中文,使用URLEncoder.encode(filePath,"UTF-8")编码后文件损坏问题得以解决,但之后我把文件名改成了时间戳,但邮件发送的文件名可以单独设置。
接下来上代码:

	/**
     * 生成Excel表格
     *
     * @param id 榜单id
     * @return
     */
public Map<String, String> putExcel(String id) {
        List<Star> starList = new ArrayList<>();
        Connection connection = DBUtil.getConnection();
        try {
            Statement statement = connection.createStatement();
            ResultSet rs = statement.executeQuery("select 查询字段 from 表 where id in (select 中间表字段 from 中间表 where 中间表字段 = " + id + ") order by 排序字段 desc");
            while (rs.next()) {
                Xx xx= new Xx();
                xx.setId(rs.getLong("id"));
                starList.add(xx);
            }
            // starList.forEach(System.out::println);
        } catch (SQLException throwables) {
            log.error(throwables.getMessage());
            throwables.printStackTrace();
        }
        try {

            StringBuilder header = new StringBuilder();
            // 查询出表头并拼接
            Statement statement = connection.createStatement();
            ResultSet rs = statement.executeQuery("select 字段 from 表名 where id = " + id);
            while (rs.next()) {
                header.append("用于发送邮件时指定邮件附件名称");
            }


            HSSFWorkbook workbook = new HSSFWorkbook();//创建Excel文件(Workbook)
            HSSFSheet sheet = workbook.createSheet("sheet1");//创建工作表(Sheet)
            // 设置表头
            HSSFRow row = sheet.createRow(0);
            HSSFCell cell = row.createCell(0);
            cell.setCellValue(header.toString());
            CellRangeAddress region = new CellRangeAddress(0, 0, 0, 4);
            sheet.addMergedRegion(region);

			// 首行合并5格单元格并设置表名

            // 设置首行内容
            row = sheet.createRow(1);
            row.createCell(0).setCellValue("id");
            row.createCell(1).setCellValue("排名");
            row.createCell(2).setCellValue("姓名");
            row.createCell(3).setCellValue("票数");
            row.createCell(4).setCellValue("生日");
            
            // 设置第一行每列标题栏

            for (int i = 0; i < starList.size(); i++) {
            	// 循环设置表内容
                Star star = starList.get(i);
                row = sheet.createRow(i + 2);
                row.createCell(0).setCellValue(xx.getId());
                row.createCell(1).setCellValue(i + 1); // 名次
                row.createCell(2).setCellValue(xx.getName());
                row.createCell(3).setCellValue(xx.getVotes());
                row.createCell(4).setCellValue(xx.getBirthday());
            }

            String filePath = "/usr/local/emailHistory/" + new Date().getTime() + ".xls";//文件路径

            FileOutputStream out = new FileOutputStream(filePath);
            workbook.write(out);//保存Excel文件
            out.close();//关闭文件流

            System.out.println("OK,榜单输出成功!");
            Map<String, String> map = new HashMap();
            map.put("filePath", filePath);
            map.put("fileName", header.toString());
            return map;
        } catch (Exception e) {
            log.error(e.getMessage());
            e.printStackTrace();
        }
        return null;
    }

至此,查询数据生成Excel表并保存到本地已完成

.
POI的Maven依赖

		<dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi</artifactId>
            <version>4.1.0</version>
        </dependency>

发邮件的功能链接https://blog.csdn.net/weixin_45952509/article/details/108963478

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

阿韩想静静

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值