0

0

如何将工时按工作日拆分并智能分配(每日上限7小时,跳过周末)

聖光之護

聖光之護

发布时间:2026-01-16 14:46:15

|

396人浏览过

|

来源于php中文网

原创

如何将工时按工作日拆分并智能分配(每日上限7小时,跳过周末)

本文介绍如何使用 pandas 对原始任务数据进行工时拆分:基于最早开始日期,将每项任务的总工时按工作日(周一至周五)均匀分配,每日最多7小时,自动跳过周末并处理跨日连续占用场景。

在项目排期或资源调度中,常需将总工时(如开发人天)合理分配到具体工作日,同时满足两项硬性约束:单日工时上限(如7小时)仅限工作日(排除周六、周日)。本教程提供一种高效、可扩展的 pandas 实现方案,无需循环遍历,完全向量化处理。

核心思路

  1. 行重复(Repeat):根据 ceil(Hours / 7) 确定每项任务需占用的工作日数量,复制对应行数;
  2. 日期递增(BusinessDay Offset):对每个任务组内重复行,按 cumcount() 顺序叠加 BusinessDay() 偏移量,确保严格跳过周末;
  3. 工时拆分(Split Logic):除最后一日外,其余日均分配满额7小时;最后一日取余数(若余数为0,则该组末尾日也分配7小时,避免出现0小时空行)。

完整实现代码

import pandas as pd
import numpy as np

# 构建初始数据
df = pd.DataFrame({
    'ID': [1, 2, 3],
    'EarliestStart': ['28.09.2023', '29.09.2023', '15.11.2023'],
    'Hours': [15.00, 5.00, 23.00]
})
df['EarliestStart'] = pd.to_datetime(df['EarliestStart'], format='%d.%m.%Y')

# 步骤1:按需重复行(向上取整:ceil(Hours/7))
out = df.loc[df.index.repeat(np.ceil(df['Hours'].div(7)))]

# 步骤2:为每组生成递增的工作日日期
n = out.groupby(level=0).cumcount()
out['PlannedStart'] = out['EarliestStart'] + n * pd.offsets.BusinessDay()

# 步骤3:提取星期名称(中文环境可加 .dt.day_name(locale='zh_CN'))
out['Weekday'] = out['PlannedStart'].dt.day_name()

# 步骤4:计算每日工时(前N-1日为7小时,最后1日为余数,余数为0则仍为7)
remainder = out.pop('Hours') % 7
out['HoursSplitted'] = np.where(
    out.index.duplicated(keep='last') | remainder.eq(0),
    7.0,
    remainder
)

# 可选:格式化日期为 DD.MM.YYYY 并转为字符串(匹配示例输出)
out['PlannedStart'] = out['PlannedStart'].dt.strftime('%d.%m.%Y')
out['EarliestStart'] = out['EarliestStart'].dt.strftime('%d.%m.%Y')
out = out.reset_index(drop=True)[['ID', 'EarliestStart', 'PlannedStart', 'Weekday', 'HoursSplitted']]

关键注意事项

  • ✅ pd.offsets.BusinessDay() 自动跳过周末和节假日(默认不含法定假日,如需支持节假日,可替换为 CustomBusinessDay(holidays=[...]));
  • ✅ groupby(level=0).cumcount() 依赖原始索引未被重置,因此 repeat() 后必须保留原始索引层级;
  • ⚠️ 若存在多任务同日“争抢”首个可用工作日(如 ID 2 应接续 ID 1 的剩余容量),本方案不模拟实时资源抢占——它按任务独立起始日+工作日偏移分配,天然规避了同一日超配(因 BusinessDay 偏移已保证日期不重叠);若需严格按全局时间线排队(如优先级调度),需引入额外的状态追踪逻辑;
  • ? HoursSplitted 列使用浮点数,如需显示为 "7,00" 格式(逗号小数点),可在最后用 out['HoursSplitted'].apply(lambda x: f"{x:.2f}".replace('.', ',')) 转换。

该方法简洁、健壮且性能优异,适用于数千行以内的常规排期场景,是 pandas 时间序列资源分配问题的典型范式解法。

Question AI
Question AI

一款基于大模型的免费的AI问答助手、总结器、AI搜索引擎

下载

相关专题

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

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

51

2025.12.04

lambda表达式
lambda表达式

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

204

2023.09.15

python lambda函数
python lambda函数

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

190

2025.11.08

Python lambda详解
Python lambda详解

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

47

2026.01.05

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

2

2026.01.16

全民K歌得高分教程大全
全民K歌得高分教程大全

本专题整合了全民K歌得高分技巧汇总,阅读专题下面的文章了解更多详细内容。

0

2026.01.16

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

10

2026.01.16

java数据库连接教程大全
java数据库连接教程大全

本专题整合了java数据库连接相关教程,阅读专题下面的文章了解更多详细内容。

33

2026.01.15

Java音频处理教程汇总
Java音频处理教程汇总

本专题整合了java音频处理教程大全,阅读专题下面的文章了解更多详细内容。

15

2026.01.15

热门下载

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

精品课程

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

共578课时 | 46.5万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

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

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