PHP如何获取闰年二月天数_PHP闰年二月天数判断方法【方法】

1次阅读

推荐使用 cal_days_in_month(cal_gregorian, 2, $year),它不依赖时间戳、规避闰年逻辑错误,兼容1–9999年;date(‘t’) 和 datetime 因时间戳限制及废弃风险不推荐。

PHP如何获取闰年二月天数_PHP闰年二月天数判断方法【方法】

php怎么用 date()cal_days_in_month() 获取二月天数

直接看结果:闰年二月是29天,平年28天。PHP不靠手算年份,而是用内置函数查日历表——cal_days_in_month() 最稳,date('t', ...) 也能用,但得注意时间戳精度。

常见错误是硬写逻辑判断闰年(比如 $year % 4 == 0),漏掉“整百年必须被400整除”这条规则,导致2100年、1900年这类年份算错。

  • cal_days_in_month(CAL_GREGORIAN, 2, $year):最推荐,底层调用系统日历库,完全规避闰年逻辑出错风险
  • date('t', mktime(0, 0, 0, 2, 1, $year)):可行,但依赖 mktime() 对无效日期(如2月30日)的自动修正,PHP 8.1+ 已标记 mktime() 为废弃,未来可能移除
  • 别用 strtotime("2024-02-01") 再格式化,strtotime() 在时区或 DST 边界下可能返回 false,导致 date('t') 算出31天这种荒谬结果

为什么 DateTime::createFromformat() 不适合查二月天数

它本意是解析字符串,不是计算月份长度。强行用 new DateTime("2024-02-01")->format('t') 看似能跑通,但本质和 date('t') 一样,绕不开时间戳构造环节;更麻烦的是,如果传入的年份超出 PHP 时间戳范围(unix timestamp 溢出,比如年份 2038 在32位系统),会静默返回1月天数(31),而不是报错。

典型翻车场景:处理历史数据(如1752年英国历法改革年)或远期规划(2150年养老金计算),DateTime 实例化失败却无提示,format('t') 还照常返回31。

立即学习PHP免费学习笔记(深入)”;

  • 只在确定年份落在 1901–2038(32位)或 1970–2038(部分旧配置)范围内才考虑 DateTime
  • 跨世纪年份一律退回 cal_days_in_month(),它不依赖时间戳,纯查格里高利历表
  • 注意 CAL_GREGORIAN常量,不是字符串,写成 'CAL_GREGORIAN' 会报 Warning: cal_days_in_month(): invalid Calendar

cal_days_in_month() 的兼容性坑点

这个函数从 PHP 4.0 就存在,看似很稳,但有两个隐藏限制:一是 windows 下某些旧版 PHP(

真正容易被忽略的是参数顺序:cal_days_in_month($calendar, $month, $year),不是 ($year, $month)。把年份放第二位会导致返回 false,而 PHP 默认不报错,后续 if 判断直接走错分支。

  • 务必检查返回值是否为 false,尤其当 $year 来自用户输入或数据库字段时(比如空字符串、负数、超大整数)
  • Windows + PHP 5.2 环境下,传入 $year = 0 可能返回 28 而不是预期的错误,linux 下则返回 false,跨平台代码必须做类型校验
  • 不用试图用它查农历二月——CAL_JEWISHCAL_CHINESE 不支持月份天数查询,会直接返回 false

简单封装一个安全的闰年二月天数函数

把校验和兜底写进函数里,比每次调用都手动 check 更可靠。重点不是多 fancy,而是堵住 NULL、字符串、负数这些常见脏数据入口。

function getFebDays(int $year): int {     if ($year < 1 || $year > 9999) {         throw new InvalidArgumentException('Year must be between 1 and 9999');     }     $days = cal_days_in_month(CAL_GREGORIAN, 2, $year);     return $days ?: 28; // fallback only for extreme edge cases (e.g. broken extension) }

这个函数没加闰年逻辑重实现,就是信任 cal_days_in_month()。如果你发现它返回了28但你知道那年是闰年,问题一定出在 PHP 编译时没启用 calendar 扩展(检查 php -m | grep calendar),而不是代码逻辑错了。

闰年规则本身不难,难的是让代码在各种 PHP 版本、系统、输入质量下都不掉链子。最省心的做法,就是把计算交给 C 层日历库,而不是在 PHP 层重复造轮子。

text=ZqhQzanResources