计算工作时间(工作日和周六不同)

Counting working hours (different for weekdays and Saturdays)

我尝试了 @ebelendez 的代码 计算两个日期之间的工作时间,但是我对如何将星期六的值设置为 3 小时感到困惑( 08:00-11:00)。例如,工作日每天的工作时间为8小时(不包括1小时休息时间),假设我想获得周四到周六的总工作时间,预期结果为19小时。

这是我所做的。有人可以帮我解决这个问题吗?

$from = '2022-04-21 07:00:00';
$to = '2022-04-23 16:00:00';

echo abs(get_working_hours($from, $to));

function get_working_hours($from,$to){
    //config
    $ini_time = [7,0]; //hr, min
    $end_time = [16,0]; //hr, min
    //date objects
    $ini = date_create($from);
    $ini_wk = date_time_set(date_create($from),$ini_time[0],$ini_time[1]);
    $end = date_create($to);
    $end_wk = date_time_set(date_create($to),$end_time[0],$end_time[1]);
    //days
    $workdays_arr = get_workdays($ini,$end);
    $workdays_count = count($workdays_arr);
    $workday_seconds = (($end_time[0] * 60 + $end_time[1]) - ($ini_time[0] * 60 + $ini_time[1])) * 60 - 3600;
    //get time difference
    $ini_seconds = 0;
    $end_seconds = 0;
    if(in_array($ini->format('Y-m-d'),$workdays_arr)) $ini_seconds = $ini->format('U') - $ini_wk->format('U');
    if(in_array($end->format('Y-m-d'),$workdays_arr)) $end_seconds = $end_wk->format('U') - $end->format('U');
    $seconds_dif = $ini_seconds > 0 ? $ini_seconds : 0;
    if($end_seconds > 0) $seconds_dif += $end_seconds;
    //final calculations
    $working_seconds = ($workdays_count * $workday_seconds) - $seconds_dif;
    return $working_seconds / 3600; //return hrs
}

function get_workdays($ini,$end){
    //config
    $skipdays = [0]; //sunday:0
    $skipdates = []; 
    //vars
    $current = clone $ini;
    $current_disp = $current->format('Y-m-d');
    $end_disp = $end->format('Y-m-d');
    $days_arr = [];
    //days range
    while($current_disp <= $end_disp){
        if(!in_array($current->format('w'),$skipdays) && !in_array($current_disp,$skipdates)){
            $days_arr[] = $current_disp;
        }
        $current->add(new DateInterval('P1D')); //adds one day
        $current_disp = $current->format('Y-m-d');
    }
    return $days_arr;
}

您的代码和链接的答案似乎不必要地复杂。我们真正需要的是:

  1. 配置每天应该计算多少小时;
  2. 创建一个可迭代的 DatePeriod(期间的每个日期都有 DateTime 个对象);
  3. 迭代日期,查找每天应该计算多少小时,总结一下。
class CountWorkingHours
{
    // Define hours counted for each day:
    public array $hours = [ 
        'Mon' => 8, 
        'Tue' => 8, 
        'Wed' => 8, 
        'Thu' => 8, 
        'Fri' => 8, 
        'Sat' => 3, 
        'Sun' => 0
    ];

    // Method for counting the hours:
    public function get_hours_for_period(string $from, string $to): int
    {
        // Create DatePeriod with requested Start/End dates:
        $period = new DatePeriod(
            new DateTime($from), 
            new DateInterval('P1D'), 
            new DateTime($to)
        );
        
        $hours = [];

        // Loop over DateTime objects in the DatePeriod:
        foreach($period as $date) {
            // Get name of day and add matching hours:
            $day = $date->format('D');
            $hours[] = $this->hours[$day];
        }
        // Return sum of hours:
        return array_sum($hours);
    }
}

使用率(returns给定时间段内工作时间的整数):

$cwh = new CountWorkingHours();
$hours = $cwh->get_hours_for_period('2022-04-21 07:00:00', '2022-04-30 16:00:00');
// = 62

如果您需要考虑 public 假期等标准每周小时数的例外情况,您可以在周期循环内添加一个“跳过日期”检查。例如,$skip_dates 属性 的数组为 non-work-days,然后在增加给定日期的工作时间之前检查 !in_array($date->format('Y-m-d'), $this->skip_dates)

P.S。此代码假定 您正在计算整个工作日 。如果您的开始或结束时间是在工作日的中间定义的,则不会被计算在内。 (如果您想将其考虑在内,则必须配置每日工作时间,并且代码必须在评估开始和结束日期时考虑到这一点。对于当前目的而言,这似乎是一项不必要的练习。)