На додаток до дуже корисної відповіді @ fyrye, це нормальне вирішення згаданої помилки ( цього ), що DatePeriod віднімає одну годину при вході в літній час, але не додає години, коли йде літній час (і, таким чином, Європа / Берлінський березень має свій правильно 743 години, але у жовтні 744 замість 745 годин):
Підрахунок годин місяця (або будь-який часовий проміжок) з урахуванням переходів DST в обох напрямках
function getMonthHours(string $year, string $month, \DateTimeZone $timezone): int
{
// or whatever start and end \DateTimeInterface objects you like
$start = new \DateTimeImmutable($year . '-' . $month . '-01 00:00:00', $timezone);
$end = new \DateTimeImmutable((new \DateTimeImmutable($year . '-' . $month . '-01 23:59:59', $timezone))->format('Y-m-t H:i:s'), $timezone);
// count the hours just utilizing \DatePeriod, \DateInterval and iterator_count, hell yeah!
$hours = iterator_count(new \DatePeriod($start, new \DateInterval('PT1H'), $end));
// find transitions and check, if there is one that leads to a positive offset
// that isn't added by \DatePeriod
// this is the workaround for https://bugs.php.net/bug.php?id=75685
$transitions = $timezone->getTransitions((int)$start->format('U'), (int)$end->format('U'));
if (2 === count($transitions) && $transitions[0]['offset'] - $transitions[1]['offset'] > 0) {
$hours += (round(($transitions[0]['offset'] - $transitions[1]['offset'])/3600));
}
return $hours;
}
$myTimezoneWithDST = new \DateTimeZone('Europe/Berlin');
var_dump(getMonthHours('2020', '01', $myTimezoneWithDST)); // 744
var_dump(getMonthHours('2020', '03', $myTimezoneWithDST)); // 743
var_dump(getMonthHours('2020', '10', $myTimezoneWithDST)); // 745, finally!
$myTimezoneWithoutDST = new \DateTimeZone('UTC');
var_dump(getMonthHours('2020', '01', $myTimezoneWithoutDST)); // 744
var_dump(getMonthHours('2020', '03', $myTimezoneWithoutDST)); // 744
var_dump(getMonthHours('2020', '10', $myTimezoneWithoutDST)); // 744
PS Якщо ви перевірите (довший) проміжок часу, який призводить до більше, ніж ці два переходи, моє вирішення не торкнеться відлічених годин, щоб зменшити потенціал смішних побічних ефектів. У таких випадках має бути реалізовано більш складне рішення. Можна було повторити всі знайдені переходи та порівняти поточний з останнім та перевірити, чи він один із DST true-> false.
strftime()та розділити різницю на 3600, але чи завжди це буде працювати? Чорт забирай, літній час!