搜索
首页后端开发php教程PHP strtotime函数用法、实现原理和源码分析_php技巧

源码位置:extdatephp_date.c

复制代码如下代码:

/* {{{ proto int strtotime(string time [, int now ])
   将日期和时间的字符串表示形式转换为时间戳 */
PHP_FUNCTION(strtotime)
{
    字符*次,*initial_ts;
    int   time_len,错误1,错误2;
    struct timelib_error_container *错误;
    长预设_ts = 0, ts;

    timelib_time *t,*现在;
    timelib_tzinfo *tzi;

    tzi = get_timezone_info(TSRMLS_C);

    if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, “sl”, ×, &time_len, &preset_ts) != FAILURE) {
        /* 我们有一个初始时间戳 */
        now = timelib_time_ctor();

       initial_ts = emalloc(25);
        snprintf(initial_ts, 24, “@%ld UTC”,preset_ts);
        t = timelib_strtotime(initial_ts, strlen(initial_ts), NULL, DATE_TIMEZONEDB, php_date_parse_tzfile_wrapper); /* 我们忽略这里的错误,因为这永远不会失败 */
        timelib_update_ts(t, tzi);
        现在->tz_info = tzi;
        现在->zone_type = TIMELIB_ZONETYPE_ID;
        timelib_unixtime2local(现在, t->sse);
        timelib_time_dtor(t);
        efree(initial_ts);
    } else if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, “s|l”, ×, &time_len, &preset_ts) != FAILURE) {
        /* 我们没有初始时间戳 */
        now = timelib_time_ctor();
        现在->tz_info = tzi;
        现在->zone_type = TIMELIB_ZONETYPE_ID;
        timelib_unixtime2local(now, (timelib_sll) time(NULL));
    } 其他 {
        RETURN_FALSE;
    }

    if (!time_len) {
        timelib_time_dtor(现在);   
        RETURN_FALSE;
    }

    t = timelib_strtotime(times, time_len, &error, DATE_TIMEZONEDB, php_date_parse_tzfile_wrapper);
    error1 = error->error_count;
    timelib_error_container_dtor(错误);
    timelib_fill_holes(t, now, TIMELIB_NO_CLONE);
    timelib_update_ts(t, tzi);
    ts = timelib_date_to_int(t, &error2);

    timelib_time_dtor(now);
    timelib_time_dtor(t);

    if (错误1 || 错误2) {
        RETURN_FALSE;
    } 其他 {
        RETURN_LONG(ts);
    }
}
/* }}} */


strtotime函数在使用strtotime(“-1month”)求上个月的今天时会出现一些状况,

因此也引出写这篇文章,全文包括以下内容:

1).strtotime函数的一些用法
2).strtotime函数的实现基本原理
3).strtotime(“-1月”)求值失败的原因

strtotime函数的一些用法

1、 strtotime(“JAN”)和strtotime(“一月”)

这两个效果是一样的,都是返回指定月份的今天,如果指定月份没有今天,则顺延到下个月。如在2011-03-31计算二月,代码:

复制代码如下代码:

echo date("Y-m-d H:i:s", strtotime("feb", strtotime("2011-03-31")));

程序会输出: 2011-03-03 00:00:00。 从表象来看,这个结果也许不一定是我们想要的,但是这也算是一种解决方案,这种方案是由什么决定的呢? strtotime函数在执行月份的计算时只计算了月份,相当于直接将月份设置为指定的月份的值,而如jan,january都会有一个对应内部数值。

2、 first关键字

first是一个辅助型的关键字,它可以与星期,天等可以指定确认值的关键字组合使用,如求2011年的第一个星期天:

复制代码 代码如下:

echo date("Y-m-d H:i:s", strtotime("second sunday", strtotime("2011-01-01"))), "
";

在PHP的源码中,对于first与星期和天的组合使用是分开的,即first day对应一个处理操作, 在最终的C实现中,天的值指定为1,即time结构中的d字段指定为1,如下代码:
复制代码 代码如下:

switch (time->relative.first_last_day_of) {
    case 1: /* first */
        time->d = 1;
        break;
    case 2: /* last */
        time->d = 0;
        time->m ;
        break;
}

3、previous和next关键字

与first类似,previous关键字可以与星期,天组合使用,表示指定时间的前一个星期几或前一天。如下所示代码:

复制代码 代码如下:

echo date("Y-m-d H:i:s", strtotime("previous sunday", strtotime("2011-02-01"))), "
";

程序会输出:2011-01-30 00:00:00
程序求2011-02-01的前一个星期天。

next关键字与previous相反,它表示下一个星期几或后一天。

4、 last关键字

last关键字既可以作为上一个,也可以作为最后一个。如求上一个星期天的日期:

复制代码 代码如下:

echo date("Y-m-d H:i:s", strtotime("last sunday", strtotime("2011-02-05"))), "
";

程序会输出: 2011-01-30 00:00:00

当程序作为最后时,其应用场景是指定日期所在月的最后一天,相当于date(“t”)的结果。如求2000年2月的最后一天:

复制代码 代码如下:

echo date("Y-m-d H:i:s", strtotime("last day", strtotime("2000-02-01"))), "
";

first、previous、last和this关键字在re文件中属于同一组。

5、 back和front关键字

这两个关键字是对一天中的小时的向前和向后操作,其调用格式如下:

复制代码 代码如下:

echo date("Y-m-d H:i:s", strtotime("back of 24", strtotime("2011-02-01"))), "
";
echo date("Y-m-d H:i:s", strtotime("front of 24", strtotime("2011-02-01"))), "
";

back表示将时间设置指定小时值的后一个小时的15分的位置。如果是24点,则算到第二天的0点15分。
front表示将时间设置指定小时值的前一个小时的45分的位置。如果是0点,则算前一天的23点45分。
上面的代码输出:2011-02-02 00:15:00 2011-02-01 23:45:00。 其中back of和front of后接的数组必须大于等于0并且小于等于24。

strtotime函数的实现基本原理
官方文档对于strtotime函数的说明是这样的:本函数预期接受一个包含美国英语日期格式的字符串并尝试将其解析为 Unix 时间戳 (自 January 1 1970 00:00:00 GMT 起的秒数),其值相对于 now 参数给出的时间,如果没有提供此参数则用系统当前时间。

这是一个标准PHP内置函数,从PHP4起就已经存在。strtotime函数是以一个扩展的方式加载进来的,在ext/date目录下有其全部实现。 作为一个标准的内置函数,其定义格式也是标准的,如下:

复制代码 代码如下:

PHP_FUNCTION(strtotime)
    //  处理输入,对于是否有第二个参数有没的处理

    //  调用相关函数,实现字符串的解析和结果计算

    //  返回结果
}


在输入处理中,先识别两个参数都存在的情况并进行处理,如果不是此种状态,则处理第二个参数不存在的情况, 如果都没有,则报错,返回FALSE。

strtotime函数的第一个参数是一个字符串,对于这个字符串,由于其复杂性,PHP使用了其词法解析一样的工具:re2c。在/ext/date/lib目录下,从parse_date.re文件我们可以看到其原始的re文件。 当用户以参数的形式传入一个字符串,此字符串将交给此程序处理,针对其字符串的不同,匹配不同的处理函数。 如strtotime(“yesterday”)调用,分析字符串时,将匹配yesterday字符串,此字符串对应函数如下:

复制代码 代码如下:

'yesterday'
{
    DEBUG_OUTPUT("yesterday");
    TIMELIB_INIT;
    TIMELIB_HAVE_RELATIVE();
    TIMELIB_UNHAVE_TIME();

    s->time->relative.d = -1;
    TIMELIB_DEINIT;
    return TIMELIB_RELATIVE;
}


这里有几个关键的结构体:
复制代码 代码如下:

typedef struct Scanner {
    int           fd;
    uchar        *lim、*str、*ptr、*cur、*tok、*pos;
    无符号 int 行,长度;
    struct timelib_error_container *errors;

    struct timelib_time *time;
    const timelib_tzdb *tzdb;
} 扫描仪;

typedef struct timelib_time {
    timelib_sll      y、m、d;     /* 年、月、日 */
    timelib_sll      h、i、s;     /* 小时、分钟、秒 */
    双           f;           /* 分数 */
    int              z;           /* GMT 偏移量(以分钟为单位)*/
    char            *tz_abbr;     /* 时区缩写(仅显示) */
    timelib_tzinfo *tz_info;     /* 时区结构 */
    有符号 int       dst;         /* 如果我们正在解析 DST 区域,则标记 */
    timelib_rel_time 相对时间;

    timelib_sll      sse;         /* 自纪元以来的秒数 */

    unsigned int   have_time、have_date、have_zone、have_relative、have_weeknr_day;

    unsigned int   sse_uptodate; /* !0 如果 sse 成员与日期/时间成员是最新的 */
    无符号整型 tim_uptodate; /* !0 如果日期/时间成员与 sse 成员是最新的 */
    无符号整数 is_localtime; /*  如果当前结构代表本地时间,则为 1;如果是 GMT,则为 0 */
    无符号整数 zone_type;    /*  1 时间偏移,
                                  *  3 时区标识符,
                                  *  2 时区缩写 */
} timelib_time;

typedef struct timelib_rel_time {
    timelib_sll y、m、d; /* 年、月、日 */
    timelib_sll h、i、s; /* 小时、分钟和秒 */

    int 工作日; /* 将这一天存储在“下周一”*/
    int weekday_behavior; /* 0: 前进时不应该计算当天; 1:当天*应该*被计算*/

    int first_last_day_of;
    int 反转; /* 差异是否应该反转 */
    timelib_sll 天; /* 包含*天数*,而不是 Y-M-D 差异 */

    timelib_special 特别;
    无符号整型有_weekday_relative,have_special_relative;
timelib_rel_time;


s->time->relative.d = -1;所表示的意思是当前时间的相对天数是-1。这只是中间词法解析的中间结果,但是最终结果是通过这些中间结果计算出来的。

strtotime(“-1月”)求值失败的原因

虽然strtotime(“-1月”)方法对于后一个月比前一个月的这种情况求值会失败,但是从其本质上来说,这并没有错。PHP这样实现也无可厚非。只是我们的需求决定了我们不能使用这种方法,因此我们称其为求值失败。

我们来看它的实现过程,由于没有第二个参数,所以程序使用默认的当前时间。第一个参数确定的是-1个月字符串,这个字符串对应的refile中的正则为:

复制代码如下代码:

reltextunit = (('sec'|'second'|'min'|'minute'|'hour'|'day'|'fortnight'|'forthnight'|'month'|'year') 's'?) | 'weeks' | daytext;

relnumber = ([ -]*[ t]*[0-9] );
relative = relnumber space? (reltextunit | 'week' );


最终relative会对应一系列操作,程序会识别出前面的-1 和后面的month字符串,month对应一种操作类型:TIMELIB_MONTH。 在此之后,根据识别出来的数字和操作类型执行操作,如下代码:
复制代码 代码如下:

case TIMELIB_MONTH:  s->time->relative.m = amount * relunit->multiplier; break;

如上代码,则是直接记录月份的相对值减一。 但是对于类似于3月31号这样的情况,2月没有31号,程序会自动将日期计算到下一个月。
声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
您如何防止与会议有关的跨站点脚本(XSS)攻击?您如何防止与会议有关的跨站点脚本(XSS)攻击?Apr 23, 2025 am 12:16 AM

要保护应用免受与会话相关的XSS攻击,需采取以下措施:1.设置HttpOnly和Secure标志保护会话cookie。2.对所有用户输入进行输出编码。3.实施内容安全策略(CSP)限制脚本来源。通过这些策略,可以有效防护会话相关的XSS攻击,确保用户数据安全。

您如何优化PHP会话性能?您如何优化PHP会话性能?Apr 23, 2025 am 12:13 AM

优化PHP会话性能的方法包括:1.延迟会话启动,2.使用数据库存储会话,3.压缩会话数据,4.管理会话生命周期,5.实现会话共享。这些策略能显着提升应用在高并发环境下的效率。

什么是session.gc_maxlifetime配置设置?什么是session.gc_maxlifetime配置设置?Apr 23, 2025 am 12:10 AM

thesession.gc_maxlifetimesettinginphpdeterminesthelifespanofsessiondata,setInSeconds.1)它'sconfiguredinphp.iniorviaini_set().2)abalanceIsiseededeedeedeedeedeedeedto to to avoidperformance andununununununexpectedLogOgouts.3)

您如何在PHP中配置会话名?您如何在PHP中配置会话名?Apr 23, 2025 am 12:08 AM

在PHP中,可以使用session_name()函数配置会话名称。具体步骤如下:1.使用session_name()函数设置会话名称,例如session_name("my_session")。2.在设置会话名称后,调用session_start()启动会话。配置会话名称可以避免多应用间的会话数据冲突,并增强安全性,但需注意会话名称的唯一性、安全性、长度和设置时机。

您应该多久再生一次会话ID?您应该多久再生一次会话ID?Apr 23, 2025 am 12:03 AM

会话ID应在登录时、敏感操作前和每30分钟定期重新生成。1.登录时重新生成会话ID可防会话固定攻击。2.敏感操作前重新生成提高安全性。3.定期重新生成降低长期利用风险,但需权衡用户体验。

如何在PHP中设置会话cookie参数?如何在PHP中设置会话cookie参数?Apr 22, 2025 pm 05:33 PM

在PHP中设置会话cookie参数可以通过session_set_cookie_params()函数实现。1)使用该函数设置参数,如过期时间、路径、域名、安全标志等;2)调用session_start()使参数生效;3)根据需求动态调整参数,如用户登录状态;4)注意设置secure和httponly标志以提升安全性。

在PHP中使用会议的主要目的是什么?在PHP中使用会议的主要目的是什么?Apr 22, 2025 pm 05:25 PM

在PHP中使用会话的主要目的是维护用户在不同页面之间的状态。1)会话通过session_start()函数启动,创建唯一会话ID并存储在用户cookie中。2)会话数据保存在服务器上,允许在不同请求间传递数据,如登录状态和购物车内容。

您如何在子域中分享会议?您如何在子域中分享会议?Apr 22, 2025 pm 05:21 PM

如何在子域名间共享会话?通过设置通用域名的会话cookie实现。1.在服务器端设置会话cookie的域为.example.com。2.选择合适的会话存储方式,如内存、数据库或分布式缓存。3.通过cookie传递会话ID,服务器根据ID检索和更新会话数据。

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

安全考试浏览器

安全考试浏览器

Safe Exam Browser是一个安全的浏览器环境,用于安全地进行在线考试。该软件将任何计算机变成一个安全的工作站。它控制对任何实用工具的访问,并防止学生使用未经授权的资源。

Atom编辑器mac版下载

Atom编辑器mac版下载

最流行的的开源编辑器

EditPlus 中文破解版

EditPlus 中文破解版

体积小,语法高亮,不支持代码提示功能

SecLists

SecLists

SecLists是最终安全测试人员的伙伴。它是一个包含各种类型列表的集合,这些列表在安全评估过程中经常使用,都在一个地方。SecLists通过方便地提供安全测试人员可能需要的所有列表,帮助提高安全测试的效率和生产力。列表类型包括用户名、密码、URL、模糊测试有效载荷、敏感数据模式、Web shell等等。测试人员只需将此存储库拉到新的测试机上,他就可以访问到所需的每种类型的列表。