Critical features for businesses often rely on dates, such as subscriptions, recurrent payments, or bookings.
As a PHP programmer, you will likely need to manipulate dates/times.
While it's generally a good practice to stick with native APIs to master your craft and understand how the language works, let's be pragmatic.
Carbon is a powerful library that extends DateTimeInterface, JsonSerializable, and other native PHP tools.
It provides useful helpers for manipulating both existing and new date objects:
// Tomorrow CarbonImmutable::now()->add(1, 'day'); // Copy instance from another API $dateTime = new \DateTime('first day of January 2025'); $carbon = Carbon::instance($dateTime); // Parsing values $date = Carbon::parse('2050-01-01 12:34:00', 'UTC');
Carbon extends the native API, but the DateTime object already provides tools for calculating dates and times:
$date = new DateTime('2024-07-23'); $date->add(new DateInterval('P10D')); echo $date->format('Y-m-d'); $date2 = new DateTime('2024-08-03'); $interval = $date->diff($date2); echo $interval->format('%R%a days');
You can pass strings to obtain specific outputs or calculate intervals and periods.
$yesterday = Carbon::yesterday(); $today = Carbon::today(); $now = Carbon::now(); $tomorrow = Carbon::tomorrow(); $futureDate = Carbon::today()->addDays(21)->format('Y-m-d'); if ($now->isEndOfDay()) { // Some code }
The library is convenient and robust for date comparisons, which are prone to various errors.
Carbon handles locales conveniently:
$now = Carbon::now()->locale('fr_FR'); echo $now->monthName;
The default PHP DateTime behavior can be misleading when adding or subtracting months. It can overflow!
While you may need that behavior at times, Carbon provides better control:
Carbon::useMonthsOverflow(false); Carbon::resetMonthsOverflow();
You can also configure overflows in each method when manipulating years or months (e.g, subMonthsNoOverflow, addMonthsWithOverflow).
Dates can be problematic in unit tests due to variability.
Carbon offers advanced capabilities for mocking, ensuring tests remain consistent and reliable.
This prevents random errors that can break your CI/CD pipelines.
Whether you use Carbon or not, there are practical ways to reduce potential bugs and inconsistencies:
Recent versions of PHP introduced better date/time exceptions, allowing you to catch unexpected inputs more consistently.
However, implementations can sometimes silence bad errors:
$dateInput = '2dsds'; $timestamp = strtotime($dateInput); echo date('Y-m-d', $timestamp); // Defaults back to 1970-01-01
The following code is not an improvement, even if it uses a DateTime object:
$dateInput = '2024-02-30'; // Invalid date (February has 28 days, 29 at most) $date = new DateTime($dateInput); echo $date->format('l \t\h\e jS F'); // Outputs "Friday the 1st March"
So, always validate dates/times:
$dateInput = '2dsds'; $timestamp = strtotime($dateInput); if (false === $timestamp) { throw new InvalidArgumentException(); } echo date('Y-m-d', $timestamp);
And, check inputs deeply:
$format = 'Y-m-d'; $dateInput = '2024-02-30'; $dateInfo = date_parse($dateInput); if (!checkdate($dateInfo['day'], $dateInfo['month'], $dateInfo['year'])) { throw new InvalidArgumentException(); } $date = DateTimeImmutable::createFromFormat($format, $dateInput); echo $date->format('l \t\h\e jS F');
DateTimeImmutable or CarbonImmutable cannot be changed after creation. Most of the time, you don't need to mutate the initial date.
If other parts of the code rely on that instance, you might introduce nasty bugs.
Immutable formats prevent such side effects and are usually better for readability and testing.
Neglecting timezones is risky:
If you don't specify the default timezone in your app, the server will determine it.
However, your application may be deployed on servers in different timezones.
You may also struggle with complexities like daylight saving time or storage formats (e.g., UTC vs. local time), which may be less painful with a library that encourages good practices.
The native API allows advanced manipulations of dates and times, while Carbon extends it beautifully.
PHP developers can leverage this additional layer to simplify calculations and improve readability.
Regardless of whether you use it, ensure you understand the default behavior of the native PHP DateTime object.
The above is the detailed content of PHP: Practical ways to escape from the hell of Dates. For more information, please follow other related articles on the PHP Chinese website!