扫码关注官方订阅号
正文
0
聖光之護
发布时间:2025-07-21 21:42:11
360人浏览过
来源于php中文网
原创
在java中,hashmap是一种基于哈希表的map实现,它提供了高效的键值对存储和检索能力。然而,hashmap的设计初衷是为了优化性能,而不是为了维护元素的插入顺序。这意味着当你向hashmap中添加键值对时,它们在内部的存储顺序是不确定的,并且可能会随着哈希冲突的解决或内部扩容而改变。
对于Excel这类结构化数据源,列的顺序通常是固定的且具有业务意义。例如,一个Excel表格可能有“姓名”、“年龄”、“城市”三列,如果使用HashMap存储每一行数据,那么读取后,Map中键值对的顺序可能是“年龄”->“姓名”->“城市”,而非原始的“姓名”->“年龄”->“城市”。当需要将这些数据写回Excel或进行依赖列顺序的进一步处理时,这种顺序的丢失就会成为一个严重的问题。
为了解决HashMap不保留插入顺序的问题,Java提供了两种特殊的Map实现:
考虑到我们的目标是保留Excel中“从左到右”的原始列顺序,LinkedHashMap是最佳选择。
下面我们将修改原始代码,将每一行数据的存储容器从HashMap替换为LinkedHashMap,以确保列顺序的保留。
最值得信赖的多语言 AI 内容检测器
立即学习“Java免费学习笔记(深入)”;
import org.apache.poi.ss.usermodel.*; import java.util.*; import java.util.stream.Collectors; public class ExcelDataReader { /** * 从Excel工作表中读取数据,并将其存储为List<Map<String, String>>。 * 每个Map代表一行数据,键为列标题,值为单元格内容。 * 使用LinkedHashMap确保列的顺序与Excel中一致。 * * @param sheet 要读取的Excel工作表对象。 * @return 包含所有行数据的List,每行数据是一个LinkedHashMap。 * 如果工作表为空,则返回一个空列表。 */ public static List<Map<String, String>> readExcelSheet(Sheet sheet) { Iterator<Row> rows = sheet.iterator(); // 检查工作表是否为空 if (!rows.hasNext()) { return Collections.emptyList(); } // 读取表头(第一行)以获取列名和它们的顺序 Row headerRow = rows.next(); List<String> columnHeaders = new ArrayList<>(); // 遍历表头单元格,获取所有非空列名 for (Cell cell : headerRow) { String headerValue = cell.getStringCellValue().trim(); // 获取并清理列名 if (!headerValue.isEmpty()) { columnHeaders.add(headerValue); } else { // 如果遇到空列名,认为后续没有有效列,停止读取表头 // 实际应用中可能需要更复杂的逻辑来处理空列名或合并单元格 break; } } List<Map<String, String>> allRowsData = new ArrayList<>(); // 遍历剩余的行(数据行) while (rows.hasNext()) { Row dataRow = rows.next(); // 使用LinkedHashMap来保证列的插入顺序与表头一致 Map<String, String> rowDataMap = new LinkedHashMap<>(); // 遍历已识别的列头,按顺序获取对应单元格的值 for (int i = 0; i < columnHeaders.size(); ++i) { String columnName = columnHeaders.get(i); // 获取单元格,如果单元格不存在则创建为空白单元格 Cell cell = dataRow.getCell(i, Row.MissingCellPolicy.CREATE_NULL_AS_BLANK); String cellValue = getCellValueAsString(cell); // 统一处理单元格值 rowDataMap.put(columnName, cellValue); } // 仅添加非空行到结果列表 // 判断行是否为空的逻辑:如果所有值都为空字符串,则认为是空行 if (!rowDataMap.values().stream().allMatch(String::isEmpty)) { allRowsData.add(rowDataMap); } } return allRowsData; } /** * 辅助方法:将单元格内容统一转换为字符串。 * 处理不同类型的单元格,避免toString()直接调用可能带来的问题。 * * @param cell 单元格对象。 * @return 单元格内容的字符串表示。 */ private static String getCellValueAsString(Cell cell) { if (cell == null) { return ""; } switch (cell.getCellType()) { case STRING: return cell.getStringCellValue().trim(); case NUMERIC: if (DateUtil.isCellDateFormatted(cell)) { // 处理日期类型 return cell.getDateCellValue().toString(); // 或者使用SimpleDateFormat格式化 } else { // 处理数字类型,避免科学计数法或精度问题 return String.valueOf(cell.getNumericCellValue()); } case BOOLEAN: return String.valueOf(cell.getBooleanCellValue()); case FORMULA: // 对于公式单元格,尝试获取其计算结果 try { return String.valueOf(cell.getNumericCellValue()); // 假设公式结果为数字 } catch (IllegalStateException e) { try { return cell.getStringCellValue().trim(); // 假设公式结果为字符串 } catch (IllegalStateException ex) { return ""; // 无法获取结果 } } case BLANK: return ""; default: return cell.toString().trim(); // 其他类型直接转字符串 } } // 示例用法 public static void main(String[] args) { // 假设你有一个Excel文件 "example.xlsx" // 这里只是一个模拟,实际使用需要引入Apache POI库并加载Workbook // Workbook workbook = new XSSFWorkbook(new FileInputStream("example.xlsx")); // Sheet sheet = workbook.getSheetAt(0); // 为了演示,我们创建一个模拟的Sheet对象 // 在实际项目中,你需要使用Apache POI加载真实的Excel文件 // 以下代码仅为概念性演示,不能直接运行,需要POI库支持 // 假设 sheet 对象已经通过 Workbook.getSheetAt(0) 或其他方式获得 // 模拟一个简单的Excel工作表 Workbook mockWorkbook = new org.apache.poi.xssf.usermodel.XSSFWorkbook(); Sheet mockSheet = mockWorkbook.createSheet("SampleData"); // 创建表头 Row header = mockSheet.createRow(0); header.createCell(0).setCellValue("column1"); header.createCell(1).setCellValue("column2"); // 创建数据行1 Row row1 = mockSheet.createRow(1); row1.createCell(0).setCellValue("value1"); row1.createCell(1).setCellValue("value2"); // 创建数据行2 Row row2 = mockSheet.createRow(2); row2.createCell(0).setCellValue("value3"); row2.createCell(1).setCellValue("value4"); List<Map<String, String>> data = readExcelSheet(mockSheet); System.out.println("读取到的数据 (LinkedHashMap 保持顺序):"); for (int i = 0; i < data.size(); i++) { Map<String, String> rowMap = data.get(i); System.out.println(i + " = " + rowMap); // 验证顺序 rowMap.forEach((key, value) -> System.out.println(" \"" + key + "\" -> " + value)); } // 预期输出: // 0 = {column1=value1, column2=value2} // "column1" -> value1 // "column2" -> value2 // 1 = {column1=value3, column2=value4} // "column1" -> value3 // "column2" -> value4 } }
代码解释:
<!-- Maven --> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>5.2.3</version> <!-- 使用最新稳定版本 --> </dependency> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>5.2.3</version> <!-- 用于.xlsx格式 --> </dependency>
通过将存储每行数据的Map实现从HashMap替换为LinkedHashMap,我们可以轻松解决Java读取Excel数据时列顺序混乱的问题。LinkedHashMap能够完美地保留键值对的插入顺序,这对于依赖列顺序的Excel数据处理至关重要。结合对表头列名的有序提取和单元格内容的健壮处理,我们可以构建出高效且准确的Excel数据读取器,为后续的数据操作提供可靠的基础。
相关文章
如何配置Java编程的Apache Shiro安全环境_Java权限管理
Apache Camel 中基于消息头条件实现消息复制的完整指南
Apache HttpClient 5 中配置重定向时保留敏感请求头的完整指南
Apache Beam Java 实现 JSON 数据按键聚合合并教程
java成品网站能二次开发吗 java成品网站二次开发可行性
相关标签:
本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
如何在页面刷新后持久化保存复选框的选中状态
2026-03-09 13:20
Go 程序中 HTTP 连接复用导致内存持续增长的解决方案
2026-03-09 13:22
Python正则表达式中字符类内连字符(-)的位置规则详解
2026-03-09 13:24
一键启动!KiriGiri发布D加密虚拟机破解启动器
2026-03-09 13:33
Python中批量替换列表中所有匹配元素的正确方法
2026-03-09 13:41
如何在 Pandas 中安全获取首个存在的列值(避免 KeyError)
2026-03-09 13:54
如何在时间类中保留原始输入值以实现准确的总秒数计算
2026-03-09 13:56
《杀戮尖塔2》Steam峰值超57万 成2026年首发最高游戏
2026-03-09 13:59
如何在单页中正确结合 AJAX 与 PHP 实现用户名实时校验
2026-03-09 14:01
壹心理如何提问
2026-03-09 14:15
热门AI工具
幻方量化公司旗下的开源大模型平台
字节跳动自主研发的一系列大型语言模型
阿里巴巴推出的全能AI助手
腾讯混元平台推出的AI助手
文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。
基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿
一站式AI创作平台,免费AI图片和视频生成。
最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。
智谱清言 - 免费全能的AI助手
相关专题
本专题聚焦 Java 主流构建工具 Maven 的学习与应用,系统讲解项目结构、依赖管理、插件使用、生命周期与多模块项目配置。通过企业管理系统、Web 应用与微服务项目实战,帮助学员全面掌握 Maven 在 Java 项目构建与团队协作中的核心技能。
2025.09.15
在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。
1010
2023.08.02
if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。
846
2023.08.22
js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。
760
2023.08.03
js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。
220
2023.09.04
java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。
1565
2023.10.24
字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。
649
2023.11.24
本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。
4
2026.03.10
热门下载
相关下载
精品课程
共22课时 | 1.8万人学习
共18课时 | 3.3万人学习
共14课时 | 3.2万人学习
共6课时 | 11.3万人学习
共79课时 | 153.8万人学习
共6课时 | 53.6万人学习
共4课时 | 22.5万人学习
共13课时 | 1.0万人学习
最新文章
微信扫码关注PHP中文网服务号
QQ扫码加入技术交流群
Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号
PHP学习
技术支持
返回顶部