0

0

Pandas DataFrame 高级合并技巧:处理共有与独有键的策略

碧海醫心

碧海醫心

发布时间:2025-09-20 10:58:28

|

506人浏览过

|

来源于php中文网

原创

Pandas DataFrame 高级合并技巧:处理共有与独有键的策略

本教程深入探讨如何在Pandas中高效合并两个DataFrame,以实现对共有键的数据进行更新和扩展,同时将独有键的数据作为新行添加。我们将详细介绍使用 DataFrame.join(how='outer') 和 DataFrame.combine_first() 两种方法,并通过实例代码展示如何处理多列键,最终生成一个包含所有信息且结构清晰的合并结果。

1. 问题背景与目标

在数据分析和处理过程中,我们经常需要将来自不同来源的数据进行整合。一个常见的需求是,当两个dataframe拥有共同的标识键时,我们希望合并它们的数据,使得:

  • 对于具有相同键的数据行,能够将第二个DataFrame中的新列添加到第一个DataFrame中,实现数据的扩展。
  • 对于只存在于其中一个DataFrame中的键,能够将其作为新行添加到最终的合并结果中,确保所有信息不丢失。

这种合并方式旨在创建一个全面的数据集,其中包含两个原始DataFrame的所有信息,通过共享键进行对齐,并用 NaN 值填充缺失的数据。

考虑以下两个示例DataFrame dfa 和 dfb:

import pandas as pd
import numpy as np

# DataFrame A
data_a = {
    'host': ['aa', 'bb', 'cc'],
    'val1': [11, 22, 33],
    'val2': [44, 55, 66]
}
dfa = pd.DataFrame(data_a)

# DataFrame B
data_b = {
    'host': ['aa', 'bb', 'dd'],
    'val1': [11, 22, 0],
    'val3': [77, 88, 99]
}
dfb = pd.DataFrame(data_b)

print("DataFrame A:")
print(dfa)
print("\nDataFrame B:")
print(dfb)

输出:

DataFrame A:
  host  val1  val2
0   aa    11    44
1   bb    22    55
2   cc    33    66

DataFrame B:
  host  val1  val3
0   aa    11    77
1   bb    22    88
2   dd     0    99

我们期望的合并结果如下,其中 ('host', 'val1') 组合是共享键:

  host  val1  val2  val3
0   aa    11  44.0  77.0
1   bb    22  55.0  88.0
2   cc    33  66.0   NaN
3   dd     0   NaN  99.0

可以看到,('aa', 11) 和 ('bb', 22) 的数据被合并,dfa 独有的 ('cc', 33) 和 dfb 独有的 ('dd', 0) 也被保留为新行。

2. 方法一:使用 DataFrame.join(how='outer')

pandas.DataFrame.join() 方法提供了一种灵活的方式来合并两个DataFrame。当使用 how='outer' 参数时,它会执行外连接操作,确保所有在两个DataFrame中出现的键都会被包含在最终结果中。对于只存在于一个DataFrame中的键,相应缺失的数据将用 NaN 填充。

关键步骤:

  1. 设置索引: 由于我们的合并基于 host 和 val1 两列的组合,需要将它们设置为DataFrame的索引。join 方法默认基于索引进行操作。
  2. 执行外连接: 使用 dfa.join(dfb, how='outer') 进行连接。
  3. 重置索引: 合并完成后,将索引重新转换回普通列,以便于后续的数据处理。

示例代码:

# 定义用于合并的键
cols_to_merge = ['host', 'val1']

# 使用 set_index 将键列设置为索引,然后执行外连接,最后重置索引
merged_df_join = dfa.set_index(cols_to_merge).join(dfb.set_index(cols_to_merge), how='outer').reset_index()

print("\n使用 DataFrame.join(how='outer') 的合并结果:")
print(merged_df_join)

输出结果:

使用 DataFrame.join(how='outer') 的合并结果:
  host  val1  val2  val3
0   aa    11  44.0  77.0
1   bb    22  55.0  88.0
2   cc    33  66.0   NaN
3   dd     0   NaN  99.0

这种方法清晰地实现了我们期望的合并逻辑,尤其适用于需要全面保留所有键值对的场景。

吉卜力风格图片在线生成
吉卜力风格图片在线生成

将图片转换为吉卜力艺术风格的作品

下载

3. 方法二:使用 DataFrame.combine_first()

pandas.DataFrame.combine_first() 方法提供了一种不同的合并策略。它会优先保留调用者DataFrame(即 dfa)中的非 NaN 值。对于 dfa 中为 NaN 的位置,它会尝试使用参数DataFrame(即 dfb)中对应位置的值进行填充。如果 dfb 中对应位置也是 NaN,则最终结果仍为 NaN。此方法在处理数据填充和合并时非常有用,尤其当一个DataFrame是主数据源,另一个是补充数据源时。

关键步骤:

  1. 设置索引: 同样,需要将 host 和 val1 列设置为DataFrame的索引,以便 combine_first 能正确地基于这些键进行对齐和合并。
  2. 执行合并填充: 使用 dfa.combine_first(dfb)。由于 combine_first 是按元素级别的操作,它会尝试填充 dfa 中的缺失值。对于 dfb 中独有的行,dfa 对应位置视为完全缺失,因此会被 dfb 的值填充。
  3. 重置索引: 合并完成后,将索引转换回普通列。

示例代码:

# 定义用于合并的键
cols_to_merge = ['host', 'val1']

# 使用 set_index 将键列设置为索引,然后执行 combine_first,最后重置索引
merged_df_combine_first = dfa.set_index(cols_to_merge).combine_first(dfb.set_index(cols_to_merge)).reset_index()

print("\n使用 DataFrame.combine_first() 的合并结果:")
print(merged_df_combine_first)

输出结果:

使用 DataFrame.combine_first() 的合并结果:
  host  val1  val2  val3
0   aa    11  44.0  77.0
1   bb    22  55.0  88.0
2   cc    33  66.0   NaN
3   dd     0   NaN  99.0

combine_first 在此场景下也能达到相同的效果,因为它本质上是在构建一个“完整”的DataFrame,其中 dfa 的数据优先,然后用 dfb 的数据来补充 dfa 中缺失的部分,包括 dfa 中不存在的行和列。

4. 两种方法比较与选择

  • DataFrame.join(how='outer'):
    • 优点: 语义清晰,直接表达了外连接的意图,即保留所有键并合并相关列。适用于需要明确指定连接类型(内连接、左连接、右连接、外连接)的场景。
    • 缺点: 如果两个DataFrame有同名但非键的列,join 会默认重命名这些列(例如 _x, _y 后缀),可能需要额外处理。
  • DataFrame.combine_first():
    • 优点: 适用于一个DataFrame作为“主”数据源,另一个作为“补充”数据源的场景。它会智能地填充 NaN 值,且不会因为非键列同名而自动重命名。
    • 缺点: 语义上更侧重于填充缺失值而非通用的连接操作,可能不如 join 或 merge 在表达连接意图上直接。

在处理本教程中描述的特定需求(合并共有键数据并添加独有键行)时,两种方法都能有效达成目标。选择哪种方法取决于个人偏好以及对代码语义的理解。通常,join(how='outer') 在表达“全面合并”的意图上更为直观。

5. 注意事项与最佳实践

  • 多列键的处理: 当合并需要基于多列的组合时,务必使用 set_index(['col1', 'col2', ...]) 将所有键列设置为索引,这是 join 和 combine_first 能够正确执行对齐操作的基础。
  • NaN 值的理解: 合并结果中出现的 NaN 值表示在某个原始DataFrame中该位置没有对应的数据。在后续分析中,可能需要对这些 NaN 值进行填充、删除或特殊处理。
  • 性能考量: 对于非常大的数据集,索引操作 (set_index) 和合并操作都可能消耗较多内存和计算资源。在实际应用中,应根据数据规模和性能要求进行测试和优化。
  • pd.merge() 的替代: 值得一提的是,pandas.merge() 函数是Pandas中最通用的合并函数,它也可以通过 how='outer' 参数实现类似的功能。例如:
    # 使用 pd.merge 实现相同效果
    merged_df_merge = pd.merge(dfa, dfb, on=cols_to_merge, how='outer')
    print("\n使用 pd.merge(how='outer') 的合并结果:")
    print(merged_df_merge)

    merge 的优势在于可以直接指定 on 参数进行列合并,而无需先 set_index 再 reset_index,代码通常更简洁。本教程重点介绍了 join 和 combine_first 作为替代方案,但 merge 也是一个非常强大的工具

6. 总结

本教程详细介绍了在Pandas中合并两个DataFrame的两种高级方法:DataFrame.join(how='outer') 和 DataFrame.combine_first(),以满足对共有键数据进行扩展并添加独有键行的需求。通过将关键列设置为索引,这两种方法都能有效地实现数据的全面整合,生成一个包含所有原始信息且结构清晰的DataFrame。理解它们的原理和适用场景,能够帮助您在数据处理任务中做出更合适的选择,从而高效地管理和分析数据。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python 时间序列分析与预测
Python 时间序列分析与预测

本专题专注讲解 Python 在时间序列数据处理与预测建模中的实战技巧,涵盖时间索引处理、周期性与趋势分解、平稳性检测、ARIMA/SARIMA 模型构建、预测误差评估,以及基于实际业务场景的时间序列项目实操,帮助学习者掌握从数据预处理到模型预测的完整时序分析能力。

82

2025.12.04

Python 数据清洗与预处理实战
Python 数据清洗与预处理实战

本专题系统讲解 Python 在数据清洗与预处理中的核心技术,包括使用 Pandas 进行缺失值处理、异常值检测、数据格式化、特征工程与数据转换,结合 NumPy 高效处理大规模数据。通过实战案例,帮助学习者掌握 如何处理混乱、不完整数据,为后续数据分析与机器学习模型训练打下坚实基础。

34

2026.01.31

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

88

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

272

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

59

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

99

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

105

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

230

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 6.1万人学习

ASP 教程
ASP 教程

共34课时 | 5.9万人学习

Vue3.x 工具篇--十天技能课堂
Vue3.x 工具篇--十天技能课堂

共26课时 | 1.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号