参考: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. (python)学习ing||类学习,@property装饰器

    class pers(): def __init__(self,hp): self._hp=hp @property def hp(self): return self._hp @hp.setter ...

  2. 下载安装JDK 和 IntelliJ IDEA 和 ActiveMq

    wget http://yun.diandaxia.com/other/jdk-8u92-linux-x64.rpm rpm -ivh jdk-8u92-linux-x64.rpm wget http ...

  3. nginx 配置mp4文件播放

    nginx 配置mp4文件播放 ​ 由于工作需要一个离线的视频播放地址,就想简单一点直接把视频文件放到nginx里面实现视频播放,但是把mp4文件放上去之后地址栏输入地址直接就是下载文件,这跟我想象的 ...

  4. NC24438 [USACO 2016 Ope P]262144

    题目链接 题目 题目描述 Bessie likes downloading games to play on her cell phone, even though she does find the ...

  5. MySQL专题2: 事务和锁

    合集目录 MySQL专题2: 事务和锁 说说数据库事务特性及存在的问题 这属于数据库事务的基础概念了, 就是ACID Atomicity, 原子性, 事务包含的所有操作要么全部成功, 要么全部失败回滚 ...

  6. 我的小程序之旅二:如何创建一个微信小程序

    第一步.准备邮箱 如果只是个人想体验一下小程序,直接用自己的QQ邮箱就行,但是这样申请的小程序很多权限都是没有的,比如获取用户手机号授权. 如果是企业或服务商要进行开发小程序,那么至少准备三个邮箱,同 ...

  7. KPTP 汇报模板

    1.什么是KPTP 它是由4个单词:Keep.Problem.Try.Plan的首字母组成的. K:keep,今天做了哪些工作: P:problem,遇到了哪些问题: T:try,计划尝试如何解决这些 ...

  8. 死锁,互斥锁,递归锁,线程事件Event,线程队列Queue,进程池和线程池,回调函数,协程的使用,协程的例子---day33

    1.死锁,互斥锁,递归锁 # ### 死锁 互斥锁 递归锁 from threading import Lock,Thread,RLock #递归锁 import time noddle_lock = ...

  9. django学习第十一天---django操作cookie和session

    Cookie cookie解析 会话 http协议是无状态的,无连接的 导致每次客户端访问服务端需要登录成功之后才能访问的页面,都需要用户再重新登录一遍,用户体验极差. 客户端想了个办法,cookie ...

  10. MIG是如何向DDR中写入数据的

    1.1    先来看看信号线的描述 我们以X16的器件为例,下面的截图来自 镁光的官方手册 https://media-www.micron.com/-/media/client/global/doc ...