参考:https://blog.csdn.net/hanyi_/article/details/118117484,https://blog.csdn.net/sunyuhua_keyboard/article/details/125482353,https://lhalcyon.com/alpine-font-issue/,https://github.com/alibaba/easyexcel/issues/1476(没字体)

首先吧,因为每周列会需要看用户使用情况报表,我这边需要把整合好的数据整理下给产品,但我比较追求便捷化,想一次性做成并导出成excel,上回直接拿导出的sql已经够难看了,这回想做个全面自动化,接着就是行动,需求是看到用户使用的多个设备的不同使用时长情况,这个其实很简单无非就是把用户跟设备一条条全查出来,挨的近些到时候直接合并并居中就好了,昨天已经做成了导出一条条数据的excel,今天就是想在原来基础上合并并居中再上线测试。

于是我开始了第一步,找了好多easyExcel合并单元格的文章最终确定了一个使用strategy策略用Excel.write方式去实现的,同时辨别如果是相同的就合并单元格

以下是CustomMergeStrategy的代码



import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.metadata.data.WriteCellData;
import com.alibaba.excel.write.handler.CellWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteTableHolder;
import org.apache.commons.lang3.StringUtils;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.ss.util.CellRangeAddress; import java.util.List; public class CustomMergeStrategy implements CellWriteHandler {
@Override
public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {
if (isHead){
//如果是表头不做处理
return;
}
//如果当前是第一行不做处理
if(relativeRowIndex==0){
return;
} //获取当前行下标,上一行下标,上一行对象,上一列对象
Sheet sheet = cell.getSheet();
int rowIndex = cell.getRowIndex();
int rowIndexPrev=rowIndex- 1;
Row row = sheet.getRow(rowIndexPrev);
Cell cellPrev = row.getCell(cell.getColumnIndex()); //得到当前单元格值和上一行单元格
// Object cellValue =cell.getCellType()== CellType.STRING?cell.getStringCellValue() : cell.getNumericCellValue();
// Object cellValuePrev =cellPrev.getCellType()==CellType.STRING?cellPrev.getStringCellValue():cellPrev.getNumericCellValue();
String cellValue =cell.getCellType()== CellType.STRING?cell.getStringCellValue() : String.valueOf(cell.getNumericCellValue());
String cellValuePrev =cellPrev.getCellType()==CellType.STRING?cellPrev.getStringCellValue():String.valueOf(cellPrev.getNumericCellValue()); System.out.println("cellValuePrev = " + cellValuePrev);
System.out.println("cellValue = " + cellValue);
//如果当前单元格和上一行单元格值相等就向下执行合并
// 这边if不能调顺序
if (StringUtils.isEmpty(cellValue) && StringUtils.isEmpty(cellValuePrev)){
return;
}
if (!cellValue.equals(cellValuePrev)) {
return;
} //获取已有策略
List<CellRangeAddress> mergedRegions = sheet.getMergedRegions(); boolean mergen=false;
for (int i = 0; i < mergedRegions.size(); i++) {
CellRangeAddress cellAddresses = mergedRegions.get(i);
if(cellAddresses.isInRange(rowIndexPrev,cell.getColumnIndex())){
sheet.removeMergedRegion(i);
cellAddresses.setLastRow(rowIndex); sheet.addMergedRegion(cellAddresses);
mergen=true;
break;
} }
if (!mergen){
CellRangeAddress cellAddresses = new CellRangeAddress(rowIndexPrev, rowIndex, cell.getColumnIndex(), cell.getColumnIndex());
sheet.addMergedRegion(cellAddresses); } } }

然后被我业务代码调用实现

EasyExcel.write("/app/static/用户统计"+dateStr+".xlsx")
.head(UserTimeLengthExcelItemDTO.class)
.registerWriteHandler(new CustomMergeStrategy())
.sheet("用户统计").doWrite(tmpUserTimeLengthExcelItemDTOList);

ok以上代码就简单实现了excel导出和合并单元格,接着我看到

这个样子有点丑啊,至少要居中吧,于是找了很多甚至还想多谢个策略类然后把add到registerWriteHandler这个里面去,不过没有成功,一直报错,继续想办法不断百度百度,都不大行,那就算了我就google看下好了哇,google出来第一个说加个注解就行了,于是我便到导出dto上面加

@HeadStyle(horizontalAlignment = HorizontalAlignmentEnum.CENTER)

在执行下果然可以,看下效果图,

发现还有点别扭啊,这个字居中是居中了,但是在底部不好看啊,继续找上下居中,查了下还是注解上加

@ContentStyle(horizontalAlignment = HorizontalAlignmentEnum.CENTER, verticalAlignment = VerticalAlignmentEnum.CENTER)//内容样式

这样子就可以了嘛,发现还有问题,空的和空的合并了,但是他们不是一个用户下的,没有截图,于是我想应该要加个判断如果是空的就不要合并,看了下逻辑他只是对如果是空就就合并做了判断,我就在上头加了个空的判断吗,ok这样子一个令人满意的报表就生成好了。

上传到线上,然后尝试生成excel,看到报错

ERROR c.c.platform.core.exception.GlobalExceptionHandler - 未知异常!原因是:ExcelGenerateException: java.lang.NoClassDefFoundError: Could not initialize class sun.awt.X11FontManager

没找到,按我在本地时候以为是文件夹不存在或者没有权限,我就去改保存的文件夹路径,前后改了3次都不成功,于是我好好看了报错内容再去google上查下发现有人说这个是没有找到字体的原因,好吧,没有字体我就装字体好吧,先apt updatealiyun真的慢等了很久终于好了,再执行命令

apt-get install -y fonts-dejavu libfreetype6-dev fontconfig

也等了好久终于好了,然后尝试继续报错,我想不应该呀,我先重启容器再说,果然我再次请求生成就可以了,好吧这样子的话我只能把当前容器再打个镜像以后就以这个新镜像跑了,想起之前有个人评论说的可能是你的镜像是silm的原因,我想我要不要看下Dockerfile里用的是啥,果然看到用的FROM openjdk:17-slim,乖乖,保底的有了,我要把这个silm删了再试试,等了一会容器起来了,在尝试生成下,成功!

md,折腾了靠一天,终于好了,拜拜,去搞scoop了

关于easyExcel导出文字合并居中和服务器导出失败踩了一天的坑的更多相关文章

  1. poi合并单元格同时导出excel

    poi合并单元格同时导出excel POI进行跨行需要用到对象HSSFSheet对象,现在就当我们程序已经定义了一个HSSFSheet对象sheet. 跨第1行第1个到第2个单元格的操作为 sheet ...

  2. springMVC中使用POI方式导出excel至客户端、服务器实例

    Apache POI是Apache软件基金会的开放源码函式库,POI提供API给Java程序对Microsoft Office格式档案读和写的功能. 这里的方法支持导出excel至项目所在服务器,或导 ...

  3. C#+OpenGL+FreeType显示3D文字(1) - 从TTF文件导出字形贴图

    C#+OpenGL+FreeType显示3D文字(1) - 从TTF文件导出字形贴图 +BIT祝威+悄悄在此留下版了个权的信息说: 最近需要用OpenGL绘制文字,这是个很费时费力的事.一般的思路就是 ...

  4. icon图标和文字整体居中在button按钮

    icon图标和文字整体居中在button按钮 icon图标和文字整体居中 一般我们常做的button按钮是文字居中 现在这个需要icon图标和文字一起居中在背景色 <a href="# ...

  5. angular2^ typescript 将 文件和Json数据 合并发送到服务器(2.服务端)

    nodejs 中使用框架 express web框架 multer 文件接受 直接贴代码了,我就不解释了 "use strict"; exports.__esModule = tr ...

  6. 怎么让Word形状里的文字上下左右居中

    怎么让Word形状里的文字上下左右居中? 第一:左右居中,用段落居中方法: 第二:上下居中,选定图形,单击鼠标右键并选择“设置形状格式”,在选项卡的“文本框”中,选择中部对齐 效果图:

  7. css垂直居中怎么设置?文字上下居中和图片垂直居中

    css 居中分css垂直居中和css水平居中,水平居中平时比较常用,这里我们主要讲css上下居中的问题.垂直居中又分为css文字上下居中和css图片垂直居中,下面我们就分别来介绍一下. css文字上下 ...

  8. 小程序canvas中文字设置居中锚点

    小程序中经常会遇到要生成图片的需求,图片一般会加上用户的头像和昵称之类的,头像只需要把腾讯域名添加到request和download列表中,使用wx.getImageInfo()就可以缓存到本地,成功 ...

  9. i8浏览器不支持placeholder属性解决办法,以及解决后,文字不居中问题

    这里想实现的效果是:设置和移除文本框默认值,如下图鼠标放到文本框中的时候,灰字消失. 1.可以用简单的方式,就是给input文本框加上onfocus属性,如下代码: <input id=&quo ...

  10. RadioButton的drawableTop图片文字不居中

    在安卓应用的开发中,一般普通应用用到最多的就是底部放一个RadioGroup实现切换的布局,今天在实现的时候,却出现了底部RadiButton的drawableTop图片及文字无法居中的情况,经过对比 ...

随机推荐

  1. 会话跟踪技术之COOKIE

    会话跟踪技术之COOKIE 一.为什么要用会话控制 我们需要我们的站点可以跟踪客户端与服务器之间的交互,保存和记忆每个用户的身份和信息. 几个疑问 我先访问A页面后访问B页面,HTTP无法知道是不是同 ...

  2. Java并发(五)----线程常见方法总结

    常见方法 方法名 static 功能说明 注意 start()   启动一个新线程,在新的线程运行 run 方法中的代码 start 方法只是让线程进入就绪,里面代码不一定立刻运行(CPU 的时间片还 ...

  3. CF1295

    A 用计算器式显示数字,可以显示 \(n\) 段.可以显示的最大数字是多少? 如果用了一个需要至少四段的数字,一定不如把这个替换成两个 \(1\) 好. 如果一共可以用偶数个,一定是全部 \(1\). ...

  4. 收集 VSCode 常用快捷键

    快速复制行 Shift + Alt + ↑/↓ 都是往下复制行,区别是:按↓复制时光标会跟着向下移动,按↑复制时光标不移动. 向上/向下移动一行 Alt + ↑/↓ 删除整行 Ctrl + Shift ...

  5. DS12C887时钟模块, STC89和STC12的代码实现

    DS12C887是时钟芯片DS12C885集成了电池和晶振的版本. 如果拆掉DS12C887的外壳, 能看到里面就是DS12C885. 功能特性 能输出世纪.年.月.日.时.分.秒等时间信息 集成电池 ...

  6. 【OpenGL ES】光影(光照与阴影)效果

    1 前言 ​ Blinn改进的冯氏光照模型 中只展示了光照效果,本文将进一步展示阴影效果. ​ 绘制阴影,需要用到深度纹理,即从光源角度看模型并绘制一张纹理图,纹理图的颜色代表了模型上的点离光源的深度 ...

  7. CSS实现渐隐渐现效果

    CSS实现渐隐渐现效果 实现渐隐渐现效果是比较常见的一种交互方式,通常的做法是控制display属性值在none和其它值之间切换,虽说功能可以实现,但是效果略显生硬,所以会有这样的需求--希望元素消失 ...

  8. thinkphp集成webuploader实战

    介绍 最近用了下thinkphp搞自己的博客,期间用到了百度的webuploader上传图片.百度出来的参考质量一言难尽,写教程没有一点追求,千篇一律的复制粘贴,某些个作者自己都没搞清楚就发文,误人又 ...

  9. Vue+SpringBoot+ElementUI实战学生管理系统-9.教师管理模块

    1.章节介绍 前一篇介绍了班级管理模块,这一篇编写教师管理模块,需要的朋友可以拿去自己定制.:) 2.获取源码 源码是捐赠方式获取,详细请QQ联系我 :)! 3.实现效果 教师列表 修改教师 4.模块 ...

  10. 溯源反制-Mysql蜜罐

    东西比较老,类似的文章网上已经很多,原理主要是通过服务端的load data动作可以主动向客户端获取文件. 看过hfish等自带的mysql蜜罐读取/etc/passwd,感觉还差点实用性.这次文章主 ...