Oprócz bardzo pomocnej odpowiedzi @ fyrye jest to dobre obejście wspomnianego błędu ( ten ), że DatePeriod odejmuje jedną godzinę przy wejściu w lato, ale nie dodaje jednej godziny przy wyjściu z lata (i dlatego marsz Europy / Berlina ma swój poprawne 743 godziny, ale październik ma 744 zamiast 745):
Liczenie godzin w miesiącu (lub w dowolnym przedziale czasowym), biorąc pod uwagę przejścia na czas letni w obu kierunkach
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 Jeśli zaznaczysz (dłuższy) okres czasu, który prowadzi do więcej niż tych dwóch przejść, moje obejście nie dotknie liczonych godzin, aby zmniejszyć potencjalne śmieszne efekty uboczne. W takich przypadkach trzeba wdrożyć bardziej skomplikowane rozwiązanie. Można iterować po wszystkich znalezionych przejściach i porównać prąd z ostatnim i sprawdzić, czy jest to jedno z DST prawda-> fałsz.
strftime()
i podziel różnicę przez 3600, ale czy to zawsze zadziała? Cholera, czas letni!