關于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
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/83909.shtml
英文地址,請注明出處:http://en.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;即可顯示出轉換…