关于easyexcel动态下拉选问题处理

前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导致的问题,也通过将下拉数据存入隐藏Sheet再引用的方式解决。

直接上代码:

package com.tlit.platform.excel.annotations;import java.lang.annotation.*;/*** @author : Doug Liang* @date : 2025/1/22 14:21* @version: 1.0* @description: 支持easyExcel的下拉选的自定义注解*/@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ExcelSelected {/*** 固定下拉内容*/String[] source() default {};}

静态的下拉选支持直接注解配置

package com.tlit.platform.excel.util;import com.tlit.platform.excel.annotations.ExcelSelected;import java.util.Optional;/*** @author : Doug Liang* @date : 2025/1/22 14:24* @version: 1.0* @description: 解析下拉选注解工具类*/
public class ResoveSelectedUtil {public static String[] resove(ExcelSelected selected){if(!Optional.ofNullable(selected).isPresent()){return null;}// 获取固定下拉信息String[] source = selected.source();if(null != source && source.length > 0){return source;}return null;}
}

动态下拉选代码

package com.tlit.platform.excel.handler;import com.alibaba.excel.write.handler.SheetWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteWorkbookHolder;
import com.tlit.platform.excel.annotations.ExcelSelected;
import com.tlit.platform.excel.util.ResoveSelectedUtil;
import org.apache.poi.hssf.usermodel.HSSFDataValidation;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.ss.util.CellRangeAddressList;import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.Map;/*** @author : Doug Liang* @date : 2025/1/22 14:33* @version: 1.0* @description: 自定义拦截器处理*/
public class SelectedWriteHandler implements SheetWriteHandler {private final Map<Integer, String[]> map;// 调整为20大小 private static final Integer LIMIT = 20;private final char[] alphabet = new char[]{'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L','M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'};public SelectedWriteHandler(Class<?> clazz) {Field[] fields = clazz.getDeclaredFields();Map<Integer, String[]> map = new HashMap<>();for (int i = 0; i < fields.length; i++) {Field field = fields[i];// 解析注解信息ExcelSelected excelSelected = field.getAnnotation(ExcelSelected.class);if (null != excelSelected) {String[] sources = ResoveSelectedUtil.resove(excelSelected);if (null != sources && sources.length > 0) {map.put(i, sources);}}}this.map = map;}/*** 如果想动态传入对应的下拉选,则只需要将key和对应的下拉选列表传入即可** @param map Map<key , value>*            value -> {"","",""}*/public SelectedWriteHandler(Map<Integer, String[]> map) {this.map = map;}@Overridepublic void beforeSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {}@Overridepublic void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {// 这里可以对cell进行任何操作Sheet sheet = writeSheetHolder.getSheet();DataValidationHelper helper = sheet.getDataValidationHelper();String dictSheetName = "字典sheet" ;Workbook workbook = writeWorkbookHolder.getWorkbook();Sheet dictSheet = workbook.createSheet(dictSheetName);workbook.setSheetHidden(1, true);// k 为存在下拉数据集的单元格下表 v为下拉数据集map.forEach((k, v) -> {// 设置下拉单元格的首行 末行 首列 末列CellRangeAddressList rangeList = new CellRangeAddressList(1, 65536, k, k);// 下拉列表约束数据DataValidationConstraint constraint;if (v.length < LIMIT) {constraint = helper.createExplicitListConstraint(v);} else {// 获取动态数据的大小int rowLen = v.length;// 设置字典sheet页的值 每一列一个字典项for (int i = 0; i < rowLen; i++) {Row row = dictSheet.getRow(i);if (row == null) {row = dictSheet.createRow(i);}row.createCell(k).setCellValue(v[i]);}String excelColumn = getExcelColumn(k);// 下拉框数据来源 eg:字典sheet!$B1:$B2String refers = dictSheetName + "!$" + excelColumn + "$1:$" + excelColumn + "$" + rowLen;// 创建可被其他单元格引用的名称Name name = workbook.createName();// 设置名称的名字name.setNameName("dict" + k);// 设置公式name.setRefersToFormula(refers);// 设置引用约束constraint = helper.createFormulaListConstraint("dict" + k);}// 设置约束DataValidation validation = helper.createValidation(constraint, rangeList);if (validation instanceof HSSFDataValidation) {validation.setSuppressDropDownArrow(false);} else {validation.setSuppressDropDownArrow(true);validation.setShowErrorBox(true);}// 阻止输入非下拉选项的值validation.setErrorStyle(DataValidation.ErrorStyle.STOP);validation.createErrorBox("提示", "此值与单元格定义格式不一致");sheet.addValidationData(validation);});}/*** 将数字列转化成为字母列** @param num 数字* @return 字母列*/private String getExcelColumn(int num) {String column;int len = alphabet.length - 1;int first = num / len;int second = num % len;if (num <= len) {column = alphabet[num] + "" ;} else {column = alphabet[first - 1] + "" ;if (second == 0) {column = column + alphabet[len] ;} else {column = column + alphabet[second - 1] ;}}return column;}}

实际引用

EasyExcel.writerSheet(sheetName).head(head).registerWriteHandler(new SelectedWriteHandler(head))// 自适应宽度.registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).build();

效果图如下:

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/pingmian/83909.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Qt】之【Get√】【Bug】通过值捕获(或 const 引用捕获)传进 lambda,会默认复制成 const

通过值捕获&#xff08;或 const 引用捕获&#xff09;传进 lambda&#xff0c;会默认复制成 const。 背景 匿名函数外部定义 QSet<QString> nameSet,需要传入匿名函数使用修改 connect(dlg, ..., [nameSet](...) {nameSet.insert(name); // ❌ 这里其实是 const QSet…

css元素的after制作斜向的删除线

<div class"price_div"></div>.price_div{position: relative; } ::after{content: ;position: absolute;left: 0;top: 50%;width: 100%;height: 2px;background: #FF186B;transform: rotate(-5deg); }

uniapp map组件的基础与实践

UniApp 中的 map 组件用于在应用中展示地图,并且支持在地图上添加标记、绘制线条和多边形等功能。以下是一些基本用法: 1. 基本结构 首先,确保你在页面的 .vue 文件中引入了 map 组件。以下是创建一个简单地图的基本代码结构: <template><view class="con…

深入理解PHP安全漏洞:文件包含与SSRF攻击全解析

深入理解PHP安全漏洞&#xff1a;文件包含与SSRF攻击全解析 前言 在Web安全领域&#xff0c;PHP应用程序的安全问题一直备受关注。本文将深入探讨两种常见的PHP安全漏洞&#xff1a;文件包含漏洞和服务器端请求伪造(SSRF)&#xff0c;帮助开发者理解漏洞原理、利用方式以及防…

MS358A 低功耗运算放大器 车规

MS358A 低功耗运算放大器 车规 产品简述 MS358A 是双通道运算放大器&#xff0c;具有低功耗、宽电源电压范围、高单位增益带宽的特性。在特定情况下&#xff0c;压摆率可以达到0.4V/μs 。每个通道的静态电流 (5V) 只有 430μA 。 MS358A输入共模范围可以到地&#xff0c;同时…

n8n + AI Agent:AI 自动化生成测试用例并支持导出 Excel

n8n + AI Agent:AI 自动化生成测试用例并支持导出 Excel 最终成果展示一、准备工作二、手把手搭建工作流第一步:创建手动触发器 (Chat Trigger)第二步:创建 AI Agent 节点第三步:为 AI Agent 植入 DeepSeek AI 模型第四步:解析AI的响应 (Code)第五步:生成Excel文件 (Conv…

5.1 HarmonyOS NEXT系统级性能调优:内核调度、I/O优化与多线程管理实战

HarmonyOS NEXT系统级性能调优&#xff1a;内核调度、I/O优化与多线程管理实战 在HarmonyOS NEXT的全场景生态中&#xff0c;系统级性能调优是构建流畅、高效应用的关键。通过内核调度精细化控制、存储与网络I/O深度优化&#xff0c;以及多线程资源智能管理&#xff0c;开发者…

​线性注意力 vs. 传统注意力:效率与表达的博弈新解

​核心结论​&#xff1a;线性注意力用计算复杂度降维换取全局建模能力&#xff0c;通过核函数和结构优化补足表达缺陷 一、本质差异&#xff1a;两种注意力如何工作&#xff1f; ​特性​传统注意力&#xff08;Softmax Attention&#xff09;线性注意力&#xff08;Linear At…

github中main与master,master无法合并到main

文章目录 遇到问题背景怎么做 遇到问题 上传 github 时候&#xff0c;发现传上去的是 master&#xff0c;但是 github 竟然还有一个 main 背景 github 采用 main 替代 master 作为主分支不是出于技术背景&#xff0c;而是出于 2020 年全球范围内兴起的 “Black Lives Matter…

使用矩阵乘法+线段树解决区间历史和问题的一种通用解法

文章目录 前言P8868 [NOIP2022] 比赛CF1824DP9990/2020 ICPC EcFinal G 前言 一般解决普通的区间历史和&#xff0c;只需要定义辅助 c h s − t ⋅ a chs-t\cdot a chs−t⋅a&#xff0c; h s hs hs是历史和&#xff0c; a a a是区间和&#xff0c; t t t是时间戳&#xff0c…

RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)

RabbitMQ 一、RabbitMQ概述 RabbitMQ RabbitMQ最初由LShift和CohesiveFT于2007年开发&#xff0c;后来由Pivotal Software Inc.&#xff08;现为VMware子公司&#xff09;接管。RabbitMQ 是一个开源的消息代理和队列服务器&#xff0c;用 Erlang 语言编写。广泛应用于各种分布…

Python Copilot【代码辅助工具】 简介

粉丝爱买鳕鱼肠深海鳕鱼肉鱼肉香肠盼盼麦香鸡味块卡乐比&#xff08;Calbee&#xff09;薯条三兄弟 独立小包美丽雅 奶茶杯一次性饮料杯好时kisses多口味巧克力糖老金磨方【黑金系列】黑芝麻丸郑新初网红郑新初烤鲜牛肉干超人毛球修剪器去球器剃毛器衣服去毛器优惠券宁之春 红黑…

VBA进度条ProgressForm1

上一章《VBA如何使用ProgressBar进度条控件》介绍了ProgressBar控件的使用方法&#xff0c;今天我给大家介绍ProgressForm1进度条的使用方法&#xff0c;ProgressForm1是集成ProgressBar控件和Label控件的窗体&#xff0c;可以同时显示进度条和百分比&#xff0c;如下图&#x…

快速部署和启动Vue3项目

快速入门Vue3 一、安装 Node.js 和 npm Vue 3 是基于 JavaScript 的框架&#xff0c;Node.js 提供了 JavaScript 运行环境&#xff0c;npm 是 Node.js 的包管理工具&#xff0c;用于安装和管理 Vue 3 及相关依赖。访问 Node.js 官方网站&#xff08;https://nodejs.org/&…

[TIP] Ubuntu 22.04 配置多个版本的 GCC 环境

问题背景 在 Ubuntu 22.04 中安装 VMware 虚拟机时&#xff0c;提示缺少 VMMON 和 VMNET 模块 编译这两个模块需要 GCC 的版本大于 12.3.0&#xff0c;而 Ubuntu 22.04 自带的 GCC 版本为 11.4.0 因此需要安装对应的 GCC 版本&#xff0c;但为了不影响其他程序&#xff0c;需…

【西门子杯工业嵌入式-4-什么是外部中断】

西门子杯工业嵌入式-4-什么是外部中断 一、中断的基本概念1. 什么是中断2. 生活中的中断示例3. MCU 中的中断机制 二、NVIC 嵌套向量中断控制器1. NVIC 简介2. NVIC 的作用3. 中断向量表 三、中断优先级机制1. 中断优先级的含义2. 抢占与响应优先级3. 优先级分组配置 四、外部中…

Blocked aria-hidden on an element because its descendant retained focus.

问题出在 Element UI 的 el-table 组件 全选功能上&#xff0c;这是一个常见的无障碍&#xff08;a11y&#xff09;问题。这个错误提示与网页 accessibility&#xff08;无障碍访问&#xff09;相关&#xff0c;涉及 aria-hidden 属性的不当使用。 问题原因分析 1. Element U…

App/uni-app 离线本地存储方案有哪些?最推荐的是哪种方案?

以下是 UniApp 离线本地存储方案的详细介绍及推荐方案分析&#xff1a; 一、UniApp 离线本地存储方案分类 1. 基于 uni.storage 系列 API&#xff08;跨端基础方案&#xff09; API 及特点&#xff1a; 提供 uni.setStorage&#xff08;异步存储&#xff09;、uni.getStorag…

数据库系统概论(十七)超详细讲解数据库规范化与五大范式(从函数依赖到多值依赖,再到五大范式,附带例题,表格,知识图谱对比带你一步步掌握)

数据库系统概论&#xff08;十七&#xff09;超详细讲解数据库规范化与五大范式&#xff08;从函数依赖到多值依赖&#xff0c;再到五大范式&#xff0c;附带例题&#xff0c;表格&#xff0c;知识图谱对比带你一步步掌握&#xff09; 前言一、为什么需要规范化1. 我们先想一个…

交互标牌——视觉货币(数字)转换器项目及源码

一、作品简介 视觉货币&#xff08;数字&#xff09;转换器是我为交互标牌创客争霸赛设计的项目&#xff0c;项目的主要功能是能将所见的数字按照设定的公式转换成新的单位量&#xff0c;这里我主要演示的是货币转换&#xff0c;直接将摄像头对准价签&#xff0c;即可显示出转换…