计算工作时间(工作日和周六不同)
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;
}
您的代码和链接的答案似乎不必要地复杂。我们真正需要的是:
- 配置每天应该计算多少小时;
- 创建一个可迭代的
DatePeriod
(期间的每个日期都有 DateTime
个对象);
- 迭代日期,查找每天应该计算多少小时,总结一下。
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。此代码假定 您正在计算整个工作日 。如果您的开始或结束时间是在工作日的中间定义的,则不会被计算在内。 (如果您想将其考虑在内,则必须配置每日工作时间,并且代码必须在评估开始和结束日期时考虑到这一点。对于当前目的而言,这似乎是一项不必要的练习。)
我尝试了 @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;
}
您的代码和链接的答案似乎不必要地复杂。我们真正需要的是:
- 配置每天应该计算多少小时;
- 创建一个可迭代的
DatePeriod
(期间的每个日期都有DateTime
个对象); - 迭代日期,查找每天应该计算多少小时,总结一下。
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。此代码假定 您正在计算整个工作日 。如果您的开始或结束时间是在工作日的中间定义的,则不会被计算在内。 (如果您想将其考虑在内,则必须配置每日工作时间,并且代码必须在评估开始和结束日期时考虑到这一点。对于当前目的而言,这似乎是一项不必要的练习。)