在实际开发中,**双重 for 循环** 是性能杀手。如果两个集合的大小分别是 M 和 N,双重循环的时间复杂度是 O(M * N)
通过使用 **Map**,我们可以将复杂度降低到 O(M + N)。这种技术通常被称为**空间换时间**。
* * *
### 1\. 业务场景举例
假设你有两个集合:
1. **账单列表 (Bill List)**:包含 `billId` 和 `amount`。
2. **详情列表 (Detail List)**:包含 `billId` 和 `remark`。
**任务**:将 `remark` 合并到对应的账单对象中。
#### **传统做法:双重 for 循环 (低效)**
每一次外部循环都要全量扫描内部循环,如果数据量各有 1 万条,就要对比 1 亿次。
Java
~~~
for (Bill bill : bills) {
for (Detail detail : details) {
if (bill.getBillId().equals(detail.getBillId())) { // 每次都要找
bill.setRemark(detail.getRemark());
break;
}
}
}
~~~
* * *
### 2\. 优化做法:Map 索引化 (高效)
**核心思想**:先遍历一遍详情列表,把它们存入一个 Map,其中 `Key` 是关联字段(billId),`Value` 是对象本身。
#### **优化步骤代码:**
Java
~~~
// 第一步:将详情列表转化为 Map (Key 是 ID,Value 是详情对象)
// 复杂度:O(N)
Map<String, Detail> detailMap = details.stream()
.collect(Collectors.toMap(Detail::getBillId, d -> d, (v1, v2) -> v1));
// 第二步:遍历账单列表,直接从 Map 中“抓取”数据
// 复杂度:O(M)
for (Bill bill : bills) {
Detail detail = detailMap.get(bill.getBillId()); // 这里的查找速度是 O(1)
if (detail != null) {
bill.setRemark(detail.getRemark());
}
}
~~~
* * *
### 3\. 性能对比表格
假设 $M=10,000$,$N=10,000$:
| **维度** | **双重 for 循环** | **Map 优化方案** | **结论** |
| --- | --- | --- | --- |
| **查找机制** | 线性扫描(一个个数) | 哈希查找(直接定位) | Map 极快 |
| **计算次数** | $10,000 \\times 10,000 = 1$ 亿次 | $10,000 + 10,000 = 2$ 万次 | **提升 5000 倍** |
| **时间复杂度** | $O(M \\times N)$ | $O(M + N)$ | 线性增长 |
| **空间消耗** | 极低 | 较高(需要内存存 Map) | 空间换时间 |
* * *
### 4\. 什么时候用这个方案?
* **数据量大时**:当两个集合的数据量都超过 100 条,建议使用 Map 优化。
* **频繁匹配时**:如果在一个复杂的算法中需要反复查找某个 ID 对应的信息。
* **内存充足时**:Map 会占用一定的堆内存,如果内存极其紧张(如嵌入式开发),需要权衡。
* * *
### 5\. 进阶技巧:分组聚合 (`groupingBy`)
有时候,一个 ID 对应多个详情(一对多),你可以使用 Java 8 的 `groupingBy`:
Java
~~~
// 将详情按 billId 分组,Map 的 Value 是一个 List
Map<String, List<Detail>> detailGroup = details.stream()
.collect(Collectors.groupingBy(Detail::getBillId));
// 使用时
List<Detail> subDetails = detailGroup.get(bill.getBillId());
~~~
### 下一步建议
这种 Map 优化的思想在 **SQL 联表查询 (Join)** 的底层实现中也经常用到。
- 环境配置
- window怎么配置java环境变量?
- SQL学习
- 字段类型
- 1、 所有可声明的字段类型
- 2、常用的可声明的字段类型
- 创建表和表内字段
- 1、整个创建表语句
- 2、设置id主键
- 3、设置业务主键,唯一索引
- 4、设置业务字段
- 5、普通索引
- 6、唯一索引与普通索引的区别
- 7、表的引擎、字符集、排序规则和注释的设置
- Java基础语法
- 数据类型
- Java中的数据类型
- LocalDate
- 常用的数据类型
- Java 常用数据类型方法
- Java中数组、list、Map、HashMap
- 如何用Map来优化那些复杂的“双重 for 循环”查询?
- Java 8 中steam()操作流
- Java中HashMap和JSON
- Java中的JSONObject
- 工具类
- Java 常用工具类
- Arrays工具类
- Java 时间工具类
- 泛型
- Java中泛型概念
- Java中的泛型容器
- 泛型参数与函数参数的区别
- 推断出泛型实参
- Lambda表达式
- 数据分层
- 异常处理
- Java8 异常处理类总结表
- MyBatis-Plus 常用异常类总结表
- Java高级特性
- Maven
- jib-maven-plugin
- 什么是Spring Boot 的 parent pom?
- maven中各个生命周期的含义
- Spring Boot
- maven与spring boot 的关系
- Java中的连接池
- Spring JDBC
- Spring JDBC的概念
- JdbcTemplate常用的方法
- Spring中Bean的概念
- Spring中的抽象,通俗解释一下
- Spring中的事物
- Spring中的事物,通俗解释一下
- Spring中的事物抽象,常见的有哪些,列举一下
- Spring中常用的事物场景有哪些,列举一下
- Spring事务管理有哪些注解?
- Spring中使用事物处理订单的案例,列举说明一下
- Spring中声明式事务、分布式事务以及编程式事务的区别,列举一下
- 配置文件
- application-properties配置文件
- Spring Boot 的启动
- spring boot项目如何启动?
- 列举一下Spring Boot的启动过程
- SpringApplication.run方法
- Spring Boot 启动时有哪些接口?
- CommandLineRunner
- Spring Boot 的常用注解
- 系统注解
- 表格:系统注解
- @Override
- @Deprecated
- @SuppressWarnnings
- 使用在类名上的注解
- 表格:使用在类名上的注解
- @RestController
- @Controller
- @Service
- @Repository
- @Component
- @Configuration
- @Resource
- @Autowired
- @RequestMapping
- @GetMapping
- @PostMapping
- @Transactional
- @Qualifier
- 使用在方法上的注解
- 表格:使用在方法上的注解
- @RequestBody
- @PathVariable
- @Bean
- @ResponseBody
- @PreAuthorize
- 其他常用注解
- 表格:其他常用注解
- @EnableAutoConfiguration
- @SpringBootApplication
- @EnableScheduling
- @EnableAsync
- @ComponentScan
- @Aspec
- @ControllerAdvice
- @ExceptionHandler
- @Value
- @ConfigurationProperties
- @EnableConfigurationProperties
- @MapperScan
- @ApiOperation
- @Produces
- Validator验证的常用注解
- spring IoC容器
- Spring IoC容器依赖注入实现方式
- 常用依赖
- RESTEasy
- resteasy简介
- RESTEasy框架(依赖)的功能和常用注解
- MyBatis
- 简介
- paginationInterceptor
- @TableName
- @TableId
- @Param
- MyBatis-Plus
- MyBatis-Plus简介
- MyBatis-Plus的工具类
- Mybatis-Plus扩展的工具类和方法
- MyBatis-Plus中最常用的工具类方法
- MyBatis-Plus 中最常用的 4 大核心工具类
- Wrapper条件构造器
- Wrapper条件构造器详解
- Wrapper条件构造器eq等方法的参数说明
- LambdaQueryWrapper与QueryWrapper
- 日期格式是否必须转换
- Lombok
- Lombok作用详解
- @Data
- @Slf4j
- @Builder
- @EqualsAndHashCode
- @Accessors
- Jackson
- Jackson简介
- @JsonFormat
- Jackson高效地在 HashMap 和 JSON 字符串之间进行相互转换
- Hutool
- Hutool简介
- hutool依赖常用的方法
- fastjson2
- fastjson2简介
- UrlBasedCorsConfigurationSource
- 生态相关
- JBoss 社区
- 支付系统
- 1. 初始化mysql数据库流程
- 2. 初始化redis数据库的流程
- 3. 初始化rabbitmq服务
- 环球置业
- 1.模块目录结构分析
- 2. DTO(数据传输层)的核心作用
- 3. VO(视图对象层)
- 4. VO(视图对象层)和 DTO 数据传输层 的区别
