12.4. Zend_Date API 概述

虽然 Zend_Date API 简单并单一,但是通过操作和操作数的丰富组合,它的设计灵活而强大。

12.4.1. Zend_Date 选项

12.4.1.1. 选择日期格式类型

有若干方法使用日期格式串,和 PHP 的 date() 类似。如果你觉得用 PHP 的日期格式指示器( specifier) 比用 ISO 格式指示器舒服, 那么你可以使用 Zend_Date::setOptions(array('format_type' => 'php')) 。 然后,对所有接受 $format 参数的函数使用 PHP 的日期格式指示器。 使用 Zend_Date::setOptions(array('format_type' => 'iso')) 来切换回只支持 ISO 日期格式的缺省模式。 所支持格式代码,参见 第 12.6.4 节 “ 使用 PHP 的 date() 格式指定器的自定义输出格式 ”

12.4.1.2. DST 和日期数学

当处理日期时,有时候需要考虑 DST(夏令时)更改的影响,一般是加上或减去一个小时。例如,在DST更改前加 上月份到日期,如果结果是在DST更改之后,那么结果将会多或少一个小时。对于边界日期,如一个月的第一天或 最后一天的午夜,添加足够的月份数来跨国日期边界会导致日期丢失一个小时并变成前面月份的最后一个小时,出现 "off by 1" 错误。 为了避免这个情况发生,通过使用 fix_dst 选项来忽略 DST 更改。当跨过夏天/冬天 DST 边界,一般要加减一个小时。 例如,如果日期的时间部分是 00:00:00, 日期算术跨过春季 DST 会导致日期比预期少一天。 因为 Zend_Date 基于时间戳,而不是带有时间组件的日历日期,时间戳丢失一个小时导致日期在日历天上比预期少一天。 为了防止这样的问题,使用选项 fix_dst,它缺省为 true,这样 DST 在日期“算术”(addMOnth(), subMonth())里无效。 当执行日期“算术“,使用 Zend_Date::setOptions(array('fix_dst' => false)) 使日期加减考虑 DST 的调整。

如果在Zend_Date实例里的当前时区设置为 UTC 或 GMT, 就不用 'fix_dst' 选项, 这是因为这两个时区和 DST 没有关系。当你又为这个实例的时区修改为不是 UTC 或 GMT 的时区,那么先前设置的 'fix_dst'选项将用于日期“算术”。

12.4.1.3. 月份计算

当从一个日期加减月份时,如果原来的日期接近月末,结果在某月的某天可能不是所期望的。 例如,当 1月31日加上1个月是,熟悉 SQL的人将以为结果是 2月28日。另一方面,熟悉 Excel 和 OpenOffice的人将认为结果是 3月3日。 这种问题只发生在这样的情况下:结果的月份里没有原来的月份的某天。 对于 ZF 的开发者来说,使用 extend_month 选项,当它为 false,就用 SQL 的方式,当它为 true,就用电子表格(如 Excel) 的方式。 extend_month 的缺省值为 false,和 SQL 方式兼容。 缺省地,Zend_Date 通过截取日期的月末部分(如果有必要)来计算月份,当原始日期指定这个月的某天超过结果月的天数时,不需要把它包含到下个月。 使用 Zend_Date::setOptions(array('extend_month' => true)); 使月份计算就像流行的电子表格程序(如 Excel)一样。

12.4.1.4.  用 Zend_Cache 加速日期本地化和标准化

通过使用 Zend_Cache 适配器,你可以加速 Zend_Date 的运算。当你使用本地化的数据,Zend_Date 的所有方法都会加速。 例如所有接受 Zend_Date::DATEZend_Date::TIME 常量的方法将会从中受益。 用 Zend_Date::setOptions(array('cache' => $adapter)); 就可给 Zend_Date 设置一个 Zend_Cache 适配器。

12.4.1.5.  用 Zend_TimeSync 同步时间戳

一般来说服务器和计算机上的时钟都不相同。Zend_TimeSync 帮助 Zend_Date 来处理这样的问题。 你可以用 Zend_Date::setOptions(array('timesync' => $timeserver)); 来设置一个时间服务器, 它将为所有 Zend_Date 的实例设置自己的实际时间戳和真实的实际时间戳之间的偏移量。使用这个选项不修改已存在的实例的时间戳。 最好的用法是在引导(bootstrap)文件里设置它。

12.4.2.  使用日期值

一旦通过生成 Zend_Date 对象格式化输入,它就携带一个时区,但在内部用标准的 UNIX 时间戳 来表示。 为了使日期能够本地化解析,必需知道时区。缺省的时区总是 GMT/UTC。用 getTimeZone()) 来检查对象的时区。 用 setTimeZone()) 来修改对象的时区。这些对象的所有处理都假定和这个时区相关。

要小心不同时区之间的对象的混合和日期部件的匹配操作,它可能会产生不期望的结果,除非运算只和时间戳相关。 除了上述的情况,Zend_Date 对象的操作会处理不同的时区,因为在 Zend_Date 初始化时日期被格式化成时间戳。

大部分方法需要一个常量来选择需要的日期的$part ,例如 Zend_Date::HOUR 。 这些常量对下面的所有函数有效,常数列表参见这里 第 12.6.2 节 “ 所有常数列表 ” 。 如果没有指定 $part,就使用 Zend_Date::TIMESTAMP。另外,也可以对 $part 使用用户指定的格式, 和 Zend_Locale_Format::getDate() 的基本机制和格式一样。 如果日期对象用明显的无效日期(如大于12的月份)来构造,那么 Zend_Date 将抛出异常,除非没有指定日期格式。例如, $partnull 或者 Zend_Date::DATES ("宽松" 格式) 。

例 12.8.  用户指定的日期输入格式

$date1 = new Zend_Date('Feb 31, 2007', null, 'en_US');
echo $date1, "\n"; // outputs "Mar 3, 2007 12:00:00 AM"

$date2 = new Zend_Date('Feb 31, 2007', Zend_Date::DATES, 'en_US');
echo $date2, "\n"; // outputs "Mar 3, 2007 12:00:00 AM"

// strictly restricts interpretation to specified format
$date3 = new Zend_Date('Feb 31, 2007', 'MM.dd.yyyy'); 
echo $date3, "\n"; // outputs "Mar 3, 2007 12:00:00 AM"

            

如果提供了可选的 $locale 参数,那么 $locale 通过给 $date 操作数替换月和星期名来消除它的歧义, 即使根据那个地方的约定来解析日期字符串 (参见 Zend_Locale_Format::getDate())。 当 $partZend_Date::DATE*Zend_Date::TIME* 常量其中之一,本地化的字符串类型的 $date 就自动格式化。 如果 $date 是包含日期的字符串,语言的地方标识符要用来解析月和星期的名字。 如果没有 $date 输入参数,那么 $locale 参数指定地方来用于本地化输出(例如,日期格式的字符串表示)。 注意 $date 输入参数可能实际有类型名 (例如 $hour 对于 addHour()),尽管它不能防止用 Zend_Date 对象作为参数。 如果没有指定 $locale ,那么当前对象的地方用于解释 $date,或选择本地化的格式来输出。

从 Zend Framework 1.7.0 开始,Zend_Date 也支持程序范围的地方的用法。你可以简单地如下设置 Zend_Locale 实例到注册表。 如果你想多次使用同一地方,用这个符号你不需要记住对每个实例手工设置地方。

// in your bootstrap file
$locale = new Zend_Locale('de_AT');
Zend_Registry::set('Zend_Locale', $locale);

// somewhere in your application
$date = new Zend_Date('31.Feb.2007');

    

12.4.3. Basic Zend_Date Operations Common to Many Date Parts

add(), sub(), compare(), get(), and set() 方法一般用来操作日期,就是操作对象里的日期。 除了 get()以外,$date 操作数对所有这些方法是必需的,它可能是 Zend_Date 实例对象、数字字符串或一个整数。 如果 $date 不是对象,这些方法就假定它是时间戳。然而,$part 操作数控制两个日期的逻辑部分,允许操作对象的日期的部件,如年或分钟, 即使 $date 包含一个长格式日期字符串,如 "December 31, 2007 23:59:59" 。 除了 compare()get() 外,操作的结果是在对象里修改日期。

例 12.9.  日期部件的操作

$date = new Zend_Date(); // $date's timestamp === time()

// changes $date by adding 12 hours
$date->add('12', Zend_Date::HOUR);
print $date;

            

下表显示了基本操作的每个组合和若干普通的日期部件的方便方法。当使用上述一般的方法时,这些方便的方法帮助我们懒惰的程序员避免在 日期部件常量 方面的打字。 方便地,它们的名称由前缀(基本操作名)和后缀(日期部件类型)的组成,如 addYear() 。 在下面列表中,能看到所有“日期部件”和“基本操作”的组合。例如,"add"操作适用于这些日期部件,包括 addDay()addYear() 等。

这些方便的方法和基本操作方法有相同的功能,但期望字符串和整数的 $date 操作数只包含表示由方便方法的后缀指示的类型的值。 这样,当$date是字符串或整数时, 这些方法的名字(如"Year" 或 "Minute")能确定 $date 操作数的单元。

12.4.3.1.  日期部件列表

表 12.1. 日期部件

日期部件 解释
时间戳 UNIX 时间戳,表示从 January 1st, 1970 00:00:00 GMT/UTC 开始的秒数。
罗马教皇日历年(如 2006)
罗马教皇日历月(1-12, 支持本地化名字)
24 小时时钟 一天的小时数(0-23),表示从一天开始算起过去的小时数。
分钟 一小时的分钟数(0-59),表示从一小时开始算起过去的分钟数。
一分钟的秒数(0-59),表示从一分钟开始算起过去的秒数。
毫秒 毫秒表示千分之一秒(0-999)。Zend_Date 支持两个另外的方法来支持小于秒的时间单位。 缺省地,Zend_Date 实例精确到毫秒,如 getFractionalPrecision() ,我们用 setFractionalPrecision($precision) 来修改精确度。 然而,因为 Zend_Date 使用 microtime(),精确度实际上限制到微秒。
如果 $date 操作数是 Zend_Date 的实例或一个数字字符串,Zend_Date::DAY_SHORT 是从 $date 里取出。 否则,根据这些文档 Zend_Date::WEEKDAY_NARROWZend_Date::WEEKDAY_NAMEZend_Date::WEEKDAY_SHORTZend_Date::WEEKDAY (假定为罗马教皇日历)的惯例尝试从天里取出。
星期 如果 $date 操作数是 Zend_Date 的实例或一个数字字符串,Zend_Date::WEEK 是从 $date 里取出。 否则产生一个异常。(假定为罗马教皇日历)
日期 如果 $date 操作数是 Zend_Date 的实例或一个数字字符串,Zend_Date::DAY_MEDIUM 是从 $date 里取出。 否则就尝试格式化 $date 字符串成 Zend_Date::DATE_MEDIUM 日期格式。 Zend_Date::DAY_MEDIUM 格式根据地方的不同而不同。
Weekday Weekdays 表示为从 0 (星期日) 到 6 (星期六). 如果 $date 操作数是 Zend_Date 的实例或一个数字字符串,Zend_Date::WEEKDAY_DIGIT 是从 $date 里取出。 否则,根据这些文档 Zend_Date::WEEKDAY_NARROWZend_Date::WEEKDAY_NAMEZend_Date::WEEKDAY_SHORTZend_Date::WEEKDAY (假定为罗马教皇日历)的惯例尝试从天里取出。
DayOfYear Zend_Date,一年中的天表示从一年开始发生的日历天数(0-365)。和其它上述单元一样,小数四舍五入到最近的整数。(假定为罗马教皇日历)
Arpa 我们支持 Arpa 日期(如 RFC 822 格式日期)。输出使用 "GMT" 或 "Local differential hours+min" 格式(见第5节的 RFC 822)。 在 PHP 5.2.2 以前,一起使用 DATE_RFC822 常量和 PHP 日期函数有时会产生 不正确的结果。 Zend_Date 的结果是正确的。例如 Mon, 31 Dec 06 23:59:59 GMT
Iso 对于输出只支持完全的 ISO 8601 日期。例如:2009-02-14T00:31:30+01:00

12.4.3.2. List of Date Operations 日期操作列表

如果 合适的常量 用于 $part 参数, 对特定日期部件,下列基本操作可用来代替方便操作。

表 12.2. 基本操作

基本操作 解释
get()

get($part = null, $locale = null)

使用 get($part) 来获取这个对象的本地化到 $locale 为格式的字符串或整数的日期的 $part 。 当使用 BCMath 扩展,对大的值返回数字字符串而不是整数。 注: 不像 get(),其它 get*() 方便方法只返回 Zend_Date 的实例, 该实例包含一个表示选择的或计算的日期/时间的日期。

set()

set($date, $part = null, $locale = null)

对从带有 $locale 地方的输入 $date 找到的部件设置当前对象的 $part 一个相应的值。

add()

add($date, $part = null, $locale = null)

加上带有 $locale 地方的 $date$part 到当前对象的日期。

sub()

sub($date, $part = null, $locale = null)

从当前对象的日期减去 加上带有 $locale 地方的 $date$part

copyPart()

copyPart($part, $locale = null)

返回一个克隆的对象,只带有拷贝到克隆的对象的日期的 $part ,克隆带有它自己的地方任意地设置为 $locale (如果指定)。

compare()

compare($date, $part = null, $locale = null)

比较 $date$part 和这个对象的时间戳,如果相等返回 0,如果对象部分更晚,返回1,否则返回 -1。


12.4.4.  比较日期

对于列表在 第 12.4 节 “Zend_Date API 概述”的日期部件,下列基本操作没有相应的方便方法。

表 12.3.  日期比较方法

方法 解释
equals()

equals($date, $part = null, $locale = null)

如果有 $locale 地方的 $date$part 和这个对象的 $part 一样,返回 true,否则返回 false。

isEarlier()

isEarlier($date, $part = null, $locale = null)

如果对象的日期的 $part 早于 有 $locale 地方的 $date$part ,返回 true。

isLater()

isLater($date, $part = null, $locale = null)

如果对象的日期的 $part 晚于有 $locale 地方的 $date$part ,返回 true。

isToday()

isToday()

测试是否今天的年、月、日匹配这个对象的日期值,使用对象的时区。

isTomorrow()

isTomorrow()

测试是否明天的年、月、日匹配这个对象的日期值,使用对象的时区。

isYesterday()

isYesterday()

测试是否昨天的年、月、日匹配这个对象的日期值,使用对象的时区。

isLeapYear()

isLeapYear()

使用 isLeapYear() 来确定当前对象是否闰年,或使用 Zend_Date::checkLeapYear($year) 来 检查 $year,它可以是字符串、整数或 Zend_Date 的实例。这个年是闰年吗?

isDate()

isDate($date, $format = null, $locale = null)

这个方法检查给定的日期是否一个真实的日期并且如果所有检查通过返回 true。 它和 php 的 checkdate() 函数一样但也检查本地化月名和扩展了 checkdate() 范围的日期。


12.4.5. 获取日期和日期部件

有若干方法支持获取和 Zend_Date 实例相关的值。

表 12.4. 日期输出方法

方法 解释
toString()

toString($format = null, $locale = null)

直接调用或通过魔术方法 __toString()toString() 根据对象地方(或可选地指定 $locale)的约定 来自动格式化日期对象的值。支持的格式代码列表,参见 第 12.6.3 节 “ 自定义 ISO 输出格式 ”

toArray()

toArray()

根据对象的地方的约定返回选择的日期的表示的数组。返回数组和 PHP 的 getdate() 函数相同并包括:

  • Number of day as 'day' (Zend_Date::DAY_SHORT)

  • Number of month as 'month' (Zend_Date::MONTH_SHORT)

  • Year as 'year' (Zend_Date::YEAR)

  • Hour as 'hour' (Zend_Date::HOUR_SHORT)

  • Minute as 'minute' (Zend_Date::MINUTE_SHORT)

  • Second as 'second' (Zend_Date::SECOND_SHORT)

  • Abbreviated timezone as 'timezone' (Zend_Date::TIMEZONE)

  • Unix timestamp as 'timestamp' (Zend_Date::TIMESTAMP)

  • Number of weekday as 'weekday' (Zend_Date::WEEKDAY_DIGIT)

  • Day of year as 'dayofyear' (Zend_Date::DAY_OF_YEAR)

  • Week as 'week' (Zend_Date::WEEK)

  • Delay of timezone to GMT as 'gmtsecs' (Zend_Date::GMT_SECS)

toValue()

toValue($part = null)

根据对象的地方的约定返回一个整数的选择的日期 $part 表示。 当 $part选择非数字值,如 Zend_Date::MONTH_NAME_SHORT ,返回 false注:这个方法调用 get() 并把结果做成 PHP 整数,如果 get() 返回一个包含对于你的系统的PHP太大的数的字符串,结果不可预计。 使用 get() 作为替换。

get()

get($part = null, $locale = null)

这个方法返回对象根据 $locale 本地化的日期的 $part,格式为字符串或整数。 更多信息参见第 12.4.3.2 节 “List of Date Operations 日期操作列表”

now()

now($locale = null)

这个方便函数和 new Zend_Date() 相同。它以Zend_Date对象返回当前日期,有 $locale


12.4.6.  使用秒的片段

Several methods support retrieving values related to a Zend_Date instance.

表 12.5.  日期输出方法

方法 解释

getFractionalPrecision()

返回秒的精度

setFractionalPrecision()

设置秒的精度

12.4.7.  日出 / 日落

有三个方法提供访问关于太阳的地理本地化信息,包括日出和日落的时间。

表 12.6.  辅助方法

方法 解释

getSunrise($location)

返回日期的日出时间

getSunset($location)

返回日期的日落时间

getSunInfo($location)

返回带有日期的太阳日期的数组