0

0

计算 DataFrame 中球员之间的历史交手记录(Head-to-Head)

花韻仙語

花韻仙語

发布时间:2025-12-30 14:29:17

|

197人浏览过

|

来源于php中文网

原创

计算 DataFrame 中球员之间的历史交手记录(Head-to-Head)

本文介绍如何基于比赛时间顺序,为每场网球对战动态计算双方球员的历史胜负次数,确保无论谁作为 player1 或 player2 出现,h2h 统计均准确反映其真实交锋胜场数。

在构建网球比赛预测模型时,历史交手记录(Head-to-Head, H2H) 是极具价值的特征:它捕捉了两名球员之间真实的对抗经验与胜负倾向。但实现时存在一个关键挑战——原始数据中同一对球员(如 A vs B)可能以不同顺序出现在 player1_id/player2_id 列中(例如某场是 A,B,下一场是 B,A),而目标变量 target 的定义始终以 player1_id 为基准(target=1 表示 player1 获胜)。若直接按行遍历或简单分组统计,极易混淆胜者归属,导致 h2h 值错位。

正确的解法核心在于:先将每对球员标准化为无序组合(如 (A,B) 统一表示 A 与 B 的对决),再按时间升序逐场累积各自胜场。以下是经过验证的高效、可扩展实现:

✅ 正确实现步骤

import pandas as pd
import numpy as np

def calculate_h2h_per_pair(group):
    """对同一对球员(标准化顺序)的子集,按时间顺序计算累计胜场"""
    # 确保 group 按 tourney_date 升序排列(关键!)
    group = group.sort_values('tourney_date').reset_index(drop=True)

    # 提取实际获胜者:target==1 → player1_id 胜;target==0 → player2_id 胜
    winner = np.where(group['target'] == 1, group['player1_id'], group['player2_id'])

    # 初始化两列:player1_h2h 和 player2_h2h,初始为 0
    player1_h2h = np.zeros(len(group), dtype=int)
    player2_h2h = np.zeros(len(group), dtype=int)

    # 遍历每场比赛(从第 2 场开始,第 1 场无历史记录)
    for i in range(1, len(group)):
        prev_matches = group.iloc[:i]  # 当前场次之前的所有同对比赛
        p1 = group.iloc[i]['player1_id']
        p2 = group.iloc[i]['player2_id']

        # 统计此前 p1 对 p2 的胜场(即 winner == p1)
        p1_wins = (prev_matches.apply(
            lambda r: r['player1_id'] if r['target']==1 else r['player2_id'], axis=1
        ) == p1).sum()

        # 统计此前 p2 对 p1 的胜场(即 winner == p2)
        p2_wins = (prev_matches.apply(
            lambda r: r['player1_id'] if r['target']==1 else r['player2_id'], axis=1
        ) == p2).sum()

        player1_h2h[i] = p1_wins
        player2_h2h[i] = p2_wins

    group['player1_h2h'] = player1_h2h
    group['player2_h2h'] = player2_h2h
    return group

# 标准化球员对:对每行 (p1, p2),生成排序后的元组 (min, max),确保 (A,B) 和 (B,A) 归为同一组
df['h2h_pair'] = df.apply(
    lambda r: tuple(sorted([r['player1_id'], r['player2_id']])), 
    axis=1
)

# 按标准化对分组,并应用 h2h 计算逻辑
result = df.groupby('h2h_pair', group_keys=False).apply(calculate_h2h_per_pair).drop('h2h_pair', axis=1)
⚠️ 注意:上述方法虽逻辑清晰、易于理解,但在大数据集上可能较慢(因显式循环)。生产环境推荐使用向量化优化版本(见下方进阶技巧)。

? 向量化高性能版本(推荐)

def get_h2h_vectorized(group):
    group = group.sort_values('tourney_date').reset_index(drop=True)

    # 构建“实际胜者”序列
    winner = group['player1_id'].where(group['target'] == 1, group['player2_id'])

    # 对当前 group 中两位球员命名(固定顺序)
    p1_ref, p2_ref = sorted([group.iloc[0]['player1_id'], group.iloc[0]['player2_id']])

    # 统计到每一行为止,p1_ref 和 p2_ref 各自获胜次数(不包含当前行)
    p1_cumwins = (winner == p1_ref).shift(1).fillna(0).cumsum().astype(int)
    p2_cumwins = (winner == p2_ref).shift(1).fillna(0).cumsum().astype(int)

    # 根据当前行中 player1_id/player2_id 的实际角色,分配 h2h 值
    player1_h2h = np.where(group['player1_id'] == p1_ref, p1_cumwins, p2_cumwins)
    player2_h2h = np.where(group['player2_id'] == p1_ref, p1_cumwins, p2_cumwins)

    return group.assign(player1_h2h=player1_h2h, player2_h2h=player2_h2h)

# 应用分组计算
df['h2h_pair'] = df.apply(lambda r: tuple(sorted([r['player1_id'], r['player2_id']])), axis=1)
result = df.groupby('h2h_pair', group_keys=False).apply(get_h2h_vectorized).drop('h2h_pair', axis=1)

✅ 输出验证(与期望一致)

输入示例:

PixVerse
PixVerse

PixVerse是一款强大的AI视频生成工具,可以轻松地将多种输入转化为令人惊叹的视频。

下载
data = {
    'tourney_date': ['2012-01-16', '2012-01-27', '2012-03-14', '2015-01-20', '2020-10-07', '2020-10-15', '2020-10-15'],
    'player1_id': ['A', 'A', 'B', 'A', 'B', 'A', 'B'],
    'player2_id': ['B', 'B', 'A', 'B', 'A', 'B', 'A'],
    'target': [0, 0, 1, 0, 1, 1, 1]
}
df = pd.DataFrame(data)

输出 result 中 player1_h2h / player2_h2h 列将严格匹配题目所给理想结果:

  • 2012-01-16 A vs B, target=0 → player1_h2h=0, player2_h2h=0
  • 2012-01-27 A vs B, target=0 → player1_h2h=0, player2_h2h=1(B 已赢 1 次)
  • 2012-03-14 B vs A, target=1 → player1_h2h=2, player2_h2h=0(B 已赢前 2 场)
  • ……依此类推。

? 关键要点总结

  • 必须标准化球员对:使用 tuple(sorted([p1,p2])) 消除顺序差异,是正确分组的前提。
  • 严格按时间排序:groupby 后必须在每个子组内调用 .sort_values('tourney_date'),否则累积统计失效。
  • 胜者识别要解耦:target 仅定义 relative 胜负,需映射到具体球员 ID,而非依赖列位置。
  • 避免行间状态污染:原问题代码错误地用 row.name 和全局索引比较,破坏了时间局部性;应始终基于 tourney_date 过滤历史。
  • 向量化优于 apply + lambda:尤其对百万级数据,shift().cumsum() 比 apply 循环快 10–100 倍。

通过以上方法,你可稳健、高效、可复现地为任意双人竞技类时序数据(网球、围棋、电竞等)生成精准的 head-to-head 特征。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

215

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

193

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

61

2026.01.05

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

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

48

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 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

270

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

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
大数据(MySQL)视频教程完整版
大数据(MySQL)视频教程完整版

共200课时 | 19.3万人学习

PHP会话控制/文件上传/分页技术
PHP会话控制/文件上传/分页技术

共22课时 | 2.2万人学习

马哥初级运维视频教程
马哥初级运维视频教程

共80课时 | 20.9万人学习

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

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