导航
电话
咨询
地图
顶部
当使用 apache poi 等库读取 excel 文件,并将每一行数据转换为一个 map 对象时,通常会遇到一个问题:java 的 hashmap 默认不保证元素的插入顺序。这意味着,即使 excel 表格中的列是按特定顺序(例如,“列1”、“列2”)排列的,当这些列名及其对应的值被放入 hashmap 后,从 hashmap 中取出时的顺序可能与原始顺序不符。
例如,原始 Excel 数据如下:
column1 column2 value1 value2 value3 value4
如果使用 HashMap 存储,结果可能变为:
0 = "column2" -> value2 "column1" -> value1 1 = "column2" -> value4 "column1" -> value3
这种无序性在需要严格保持列顺序的场景(如重新写入 Excel、数据比对或特定业务逻辑处理)下,会带来极大的不便甚至错误。
为了解决 HashMap 的无序问题,Java 提供了 LinkedHashMap。LinkedHashMap 是 HashMap 的子类,它通过维护一个双向链表来记录元素的插入顺序。这意味着,当您向 LinkedHashMap 中添加键值对时,它们会按照添加的顺序被保留。当您遍历 LinkedHashMap 时,元素将按照它们被插入的顺序返回。
对于需要对键进行自然排序或自定义排序的场景,TreeMap 也是一个选择。TreeMap 会根据键的自然顺序或提供的 Comparator 进行排序。然而,对于保持原始的、非字母顺序的列顺序,LinkedHashMap 是更直接和高效的选择。
以下是如何修改现有的 Excel 读取方法,以使用 LinkedHashMap 来保持列的原始顺序:
import org.apache.poi.ss.usermodel.*; import java.util.*; import java.util.stream.Collectors; public class ExcelReader { /** * 从给定的 Excel Sheet 中读取数据,并将其存储为 List>。 * 每个 Map 保持原始列的插入顺序。 * * @param sheet 要读取的 Excel Sheet 对象。 * @return 包含所有行数据的 List,每行数据是一个保持列顺序的 Map。 * 如果 Sheet 为空,则返回一个空列表。 */ public static List> readExcelSheetOrdered(Sheet sheet) { Iterator rows = sheet.iterator(); // 检查 Sheet 是否为空 if (!rows.hasNext()) { return Collections.emptyList(); } // 读取标题行 Row header = rows.next(); List keys = new ArrayList<>(); // 遍历标题单元格,获取列名。 // 注意:这里假设列名是连续的,遇到空列名则停止。 for (Cell cell : header) { String value = cell.getStringCellValue().trim(); // 获取并修剪列名 if (!value.isEmpty()) { keys.add(value); } else { // 如果遇到空列名,则认为后续没有有效列,停止读取标题 break; } } // 存储所有行数据的列表 List> result = new ArrayList<>(); // 遍历数据行 while (rows.hasNext()) { Row row = rows.next(); // 使用 LinkedHashMap 替代 HashMap,以保持列的插入顺序 Map rowMap = new LinkedHashMap<>(); // 遍历每一列,根据标题行的顺序填充数据 for (int i = 0; i < keys.size(); ++i) { // 获取单元格,如果单元格不存在则创建为空白单元格 Cell cell = row.getCell(i, Row.MissingCellPolicy.CREATE_NULL_AS_BLANK); String value; // 将单元格内容转换为字符串 value = getCellValueAsString(cell); rowMap.put(keys.get(i), value); } // 仅添加非空行(即所有值不全为空的行) // 使用 stream() 和 allMatch() 检查所有值是否都为空字符串 if (!rowMap.values().stream().allMatch(String::isEmpty)) { result.add(rowMap); } } return result; } /** * 辅助方法:将单元格内容转换为字符串。 * 处理不同类型的单元格数据。 * * @param cell 单元格对象。 * @return 单元格内容的字符串表示。 */ private static String getCellValueAsString(Cell cell) { if (cell == null) { return ""; } CellType cellType = cell.getCellType(); switch (cellType) { case STRING: return cell.getStringCellValue().trim(); case NUMERIC: // 对于日期类型,需要特殊处理 if (DateUtil.isCellDateFormatted(cell)) { return cell.getDateCellValue().toString(); // 或者使用 SimpleDateFormat 格式化 } else { // 对于数字,避免科学计数法,转换为普通字符串 DataFormatter formatter = new DataFormatter(); return formatter.formatCellValue(cell); } case BOOLEAN: return String.valueOf(cell.getBooleanCellValue()); case FORMULA: // 尝试评估公式结果 try { return String.valueOf(cell.getNumericCellValue()); // 假设公式结果是数字 } catch (IllegalStateException e) { return cell.getStringCellValue().trim(); // 否则尝试作为字符串 } case BLANK: return ""; default: return cell.toString().trim(); } } // 示例用法 (假设您有一个 Workbook 对象) public static void main(String[] args) throws Exception { // 这是一个伪代码示例,实际使用需要加载 Excel 文件 // Workbook workbook = new XSSFWorkbook(new FileInputStream("your_excel_file.xlsx")); // Sheet sheet = workbook.getSheetAt(0); // List> orderedData = readExcelSheetOrdered(sheet); // 模拟一个 Sheet 对象进行测试 // 假设 sheet 已经包含数据 // ... (省略创建模拟 Sheet 的复杂代码,实际项目中通过文件读取) // 例如: // Sheet mockSheet = new XSSFWorkbook().createSheet("Test Sheet"); // Row headerRow = mockSheet.createRow(0); // headerRow.createCell(0).setCellValue("column1"); // headerRow.createCell(1).setCellValue("column2"); // // Row dataRow1 = mockSheet.createRow(1); // dataRow1.createCell(0).setCellValue("value1"); // dataRow1.createCell(1).setCellValue("value2"); // // Row dataRow2 = mockSheet.createRow(2); // dataRow2.createCell(0).setCellValue("value3"); // dataRow2.createCell(1).setCellValue("value4"); // // List> orderedData = readExcelSheetOrdered(mockSheet); // orderedData.forEach(map -> { // map.entrySet().forEach(entry -> System.out.println(entry.getKey() + " -> " + entry.getValue())); // System.out.println("---"); // }); // workbook.close(); } }
代码改动点说明:
通过将 HashMap 替换为 LinkedHashMap,可以有效地解决在 Java 中读取 Excel 数据时列顺序混乱的问题。LinkedHashMap 保证了元素的插入顺序,使得从 Excel 读取的数据能够精确地反映源文件的列布局。这对于后续的数据处理、数据验证以及将数据回写到 Excel 等操作至关重要,确保了数据流的完整性和可预测性。结合健壮的单元格类型处理,您可以构建一个可靠且高效的 Excel 数据读取模块。
# ai # switch # 是一个 # red # excel # 排列 # 对象 # Java # String # 自定义 # 子类 # 字符串 # 为空 # 不存在 # 键值对 # 数据结构 # 数据类型 # map # 键值 # 遍历 # 行数 # apache # 转换为 # 类型转换 # 单元格 # Excel 表格 # 当您
相关栏目: 【 行业资讯 】 【 网络运营 】 【 GEO优化 】 【 营销推广 】 【 SEO优化 】 【 技术教程 】 【 代码知识 】 【 AI推广 】
相关推荐: Linux如何使用Curl发送请求_Linux下API接口测试与文件下载技巧【步骤】 如何使用Golang实现容器安全扫描_Golang Docker镜像漏洞检测方法 php打包exe如何加密代码_防反编译保护方法【技巧】 Win11 C盘满了怎么清理 Win11磁盘清理和存储感知使用教程【新手必看】 C++中的协变与逆变是什么?C++函数指针与返回类型详解【类型系统】 如何使用Golang管理模块版本_Golanggo mod tidy与升级方法 Go 语言标准库为何不提供泛型 Contains 方法:设计哲学与类型系统约束 Win10如何更改开机密码_Windows10登录选项更改密码 Win10怎样卸载iTunes_Win10卸载iTunes步骤【步骤】 Win10怎么关闭自动更新错误重启 Win10策略禁止失败补丁强制重启【防护】 Win10怎么设置开机密码_Windows10账户登录密码设置与取消 Win11怎么设置任务栏大小_Windows11注册表修改TaskbarSi值 Win11怎么设置指纹解锁 Win11笔记本录入指纹登录【教程】 Win11怎么关闭系统提示音_Windows11声音方案设置为静音 mac怎么安装字体_MAC添加第三方字体与字体册管理【教程】 Win11更新后变慢怎么办_Win11系统更新后卡顿优化方案【详解】 如何在Golang中实现微服务负载均衡_Golang负载均衡策略与实现示例 Win10如何关闭安全中心所有通知 Win10禁用Windows Defender提醒【设置】 Win11怎么清理C盘OneDrive缓存_Win11清理OneDrive缓存技巧【方法】 如何在Golang中验证模块完整性_Golanggo.sum校验与安全实践 Win11怎么调整屏幕亮度_Windows 11调节显示器亮度护眼设置【步骤】 win11怎么关闭用户账户控制UAC Win11调整系统安全提示等级【详解】 Django 测试数据库表缺失与字段未创建问题的完整解决方案 如何使用Golang处理网络超时错误_Golang请求超时异常处理方法 Win11怎么设置虚拟内存_Windows 11优化内存性能提升速度【技巧】 Win11文件夹预览图不显示怎么办_Win11缩略图缓存重建修复【教程】 如何使用Golang捕获并记录协程panic_保证主程序稳定运行 PHP怎么接收URL中的锚点参数_获取#后面参数值的技巧【详解】 Win11怎么清理C盘下载文件夹_Win11清理下载文件夹技巧【教程】 LINUX如何开放防火墙端口_Linux firewalld与iptables开放端口命令【安全配置】 Win11怎么设置快速访问_Windows11文件资源管理器主页 php怎么捕获异常_trycatch结构处理运行时错误的技巧【方法】 Mac怎么设置鼠标滚动速度_Mac鼠标设置详细参数 Win11怎么关闭防火墙通知_屏蔽Win11安全中心安全警告弹窗【技巧】 Win11如何开启telnet服务 Win11启用Telnet客户端【步骤】 Win11怎么设置默认邮件客户端 Win11修改Mail应用关联【教程】 Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】 Win11色盲模式怎么开_Win11屏幕颜色滤镜设置【关怀】 Win11怎么更改电脑密码_Windows 11修改本地账户密码【步骤】 如何使用Golang操作指针变量_Golang解引用与赋值实践 Windows怎样拦截QQ浏览器广告_Windows拦截QQ浏览器广告方法【方法】 Win11怎么设置开机密码_Windows11账户登录选项PIN码 Flask 表单数据通过 SMTP 发送邮件的完整实现教程 VSC怎么创建PHP项目_从零开始搭建项目的步骤【操作】 Win11开机Logo怎么换_Win11自定义启动画面工具【高级】 Win11怎么关闭系统提示音_Windows11声音方案设为无声教程 Win11怎么更改任务栏颜色_Windows11个性化重音色设置 Mac上的iMovie如何剪辑视频?(新手入门教程) php嵌入式需要什么环境_搭建php+linux嵌入式开发环境【详解】 如何使用Golang实现多重错误处理_Golangerror组合与判断方法
赣ICP备2024031479号