资讯专栏INFORMATION COLUMN

set_limit_time()、ini_set()解析

Atom / 2397人阅读

摘要:问题发现今天写了一个脚本,提交代码的时候京哥给我,果断帮我指出这个脚本的运行时间限制不要这么写要这么写然后给我讲了一堆原理,什么直接进内存啊,需要暂时修改原配置啊恩,还是高工懂得多,于是我开始对两个函数进行了测试。

问题发现

今天写了一个脚本,提交代码的时候京哥给我cr,果断帮我指出这个脚本的运行时间限制不要这么写

</>复制代码

  1. ini_set("max_execution_time","30");

要这么写

</>复制代码

  1. set_limit_time(30);

然后给我讲了一堆原理,什么 set_limit_time() 直接进内存啊,ini_set("max_execution_time",) 需要暂时修改原配置啊...恩,还是高工懂得多,于是我开始对两个函数进行了测试。

测试

测试代码如下:

</>复制代码

  1. 这不测不要紧,一测就发现了问题,两次测试都是先sleep10s,然后返回

  2. </>复制代码

    1. beginFatal error: Maximum execution time of 1 second exceeded in /Users/jdq/test.php on line 6
  3. 难道sleep不算脚本执行的时间?答案应该是肯定的,可是我以前测试后端接口超时的时候确实用的sleep,而且也超时返回了504,思考了一下应该是php-fpm的配置覆盖了php的ini配置的原因吧,所以sleep的时间也视为一个cgi进程的执行时间。(此处推断有待确定)回归正题,马上修改了测试代码

  4. </>复制代码

    1. 两个脚本都是执行了1s,直接fatal。那这两个函数又是在什么阶段起作用的呢,修改测试代码为

    2. </>复制代码

      1. 返回结果分别为

      2. </>复制代码

        1. ini_set结果:
        2. 1528297536
        3. begin
        4. Fatal error: Maximum execution time of 5 seconds exceeded in /Users/jdq/test.php on line 11
        5. 1528297546
        6. set_time_limit结果:
        7. 1528297751
        8. begin
        9. Fatal error: Maximum execution time of 5 seconds exceeded in /Users/jdq/test.php on line 11
        10. 1528297761
      3. 这两个函数都是在执行的时候才开始限定脚本执行时间,感觉并没有什么区别,所以我找了找两个函数的源码。

      4. 函数源码
      5. set_limit_time()
      6. 源码如下(以下均为php7.1源码)

      7. </>复制代码

        1. PHP_FUNCTION(set_time_limit)
        2. {
        3. zend_long new_timeout;
        4. char *new_timeout_str;
        5. int new_timeout_strlen;
        6. zend_string *key;
        7. //做了一些参数校验
        8. if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &new_timeout) == FAILURE) {
        9. return;
        10. }
        11. new_timeout_strlen = (int)zend_spprintf(&new_timeout_str, 0, ZEND_LONG_FMT, new_timeout);
        12. //看到配置项max_execution_time这里我心里就开始哈哈哈了
        13. key = zend_string_init("max_execution_time", sizeof("max_execution_time")-1, 0);
        14. //其实调用了zend_alter_ini_entry_chars_ex这个函数
        15. if (zend_alter_ini_entry_chars_ex(key, new_timeout_str, new_timeout_strlen, PHP_INI_USER, PHP_INI_STAGE_RUNTIME, 0) == SUCCESS) {
        16. RETVAL_TRUE;
        17. } else {
        18. RETVAL_FALSE;
        19. }
        20. zend_string_release(key);
        21. efree(new_timeout_str);
        22. }
      8. 通过源码我们可以看出,set_limit_time 函数就是调用了 zend_alter_ini_entry_chars_ex 对配置项 max_execution_time 进行了一番操作,这个函数的源代码

      9. </>复制代码

        1. ZEND_API int zend_alter_ini_entry_chars_ex(zend_string *name, const char *value, size_t value_length, int modify_type, int stage, int force_change) /* {{{ */
        2. {
        3. int ret;
        4. zend_string *new_value;
        5. new_value = zend_string_init(value, value_length, !(stage & ZEND_INI_STAGE_IN_REQUEST));
        6. //执行了zend_alter_ini_entry_ex这个函数
        7. ret = zend_alter_ini_entry_ex(name, new_value, modify_type, stage, force_change);
        8. zend_string_release(new_value);
        9. return ret;
        10. }
      10. 所以可以看出,set_limit_time 最终实现要是 zend_alter_ini_entry_ex ,下面我们将讨论这个函数。

      11. ini_set
      12. 源码如下

      13. </>复制代码

        1. PHP_FUNCTION(ini_set)
        2. {
        3. zend_string *varname;
        4. zend_string *new_value;
        5. zend_string *val;
        6. //参数处理
        7. ZEND_PARSE_PARAMETERS_START(2, 2)
        8. Z_PARAM_STR(varname)
        9. Z_PARAM_STR(new_value)
        10. ZEND_PARSE_PARAMETERS_END();
        11. //去一张hash表根据配置项名字寻找当前value,下面会说到,这个value通常会被释放掉
        12. val = zend_ini_get_value(varname);
        13. /* copy to return here, because alter might free it! */
        14. if (val) {
        15. if (ZSTR_IS_INTERNED(val)) {
        16. RETVAL_INTERNED_STR(val);
        17. } else if (ZSTR_LEN(val) == 0) {
        18. RETVAL_EMPTY_STRING();
        19. } else if (ZSTR_LEN(val) == 1) {
        20. RETVAL_INTERNED_STR(ZSTR_CHAR((zend_uchar)ZSTR_VAL(val)[0]));
        21. } else if (!(GC_FLAGS(val) & GC_PERSISTENT)) {
        22. ZVAL_NEW_STR(return_value, zend_string_copy(val));
        23. } else {
        24. ZVAL_NEW_STR(return_value, zend_string_init(ZSTR_VAL(val), ZSTR_LEN(val), 0));
        25. }
        26. } else {
        27. RETVAL_FALSE;
        28. }
        29. //一堆我也不知道要干什么的校验
        30. #define _CHECK_PATH(var, var_len, ini) php_ini_check_path(var, var_len, ini, sizeof(ini))
        31. /* open basedir check */
        32. if (PG(open_basedir)) {
        33. if (_CHECK_PATH(ZSTR_VAL(varname), ZSTR_LEN(varname), "error_log") ||
        34. _CHECK_PATH(ZSTR_VAL(varname), ZSTR_LEN(varname), "java.class.path") ||
        35. _CHECK_PATH(ZSTR_VAL(varname), ZSTR_LEN(varname), "java.home") ||
        36. _CHECK_PATH(ZSTR_VAL(varname), ZSTR_LEN(varname), "mail.log") ||
        37. _CHECK_PATH(ZSTR_VAL(varname), ZSTR_LEN(varname), "java.library.path") ||
        38. _CHECK_PATH(ZSTR_VAL(varname), ZSTR_LEN(varname), "vpopmail.directory")) {
        39. if (php_check_open_basedir(ZSTR_VAL(new_value))) {
        40. zval_dtor(return_value);
        41. RETURN_FALSE;
        42. }
        43. }
        44. }
        45. #undef _CHECK_PATH
        46. //最终要执行zend_alter_ini_entry_ex这个函数
        47. if (zend_alter_ini_entry_ex(varname, new_value, PHP_INI_USER, PHP_INI_STAGE_RUNTIME, 0) == FAILURE) {
        48. zval_dtor(return_value);
        49. RETURN_FALSE;
        50. }
        51. }
      14. ini_set 的实现也是要靠函数 zend_alter_ini_entry_ex ,而 set_limit_time 只是其中一个配置项(参数)为 max_execution_time 的实现而已,原来这两个函数实现机制是一样的,这时京哥的脸色已经发生了微微的变化,哈哈哈......

      15. 那么zend_alter_ini_entry_ex又是如何实现的呢?

      16. zend_alter_ini_entry_ex
      17. 源码如下

      18. </>复制代码

        1. ZEND_API int zend_alter_ini_entry_ex(zend_string *name, zend_string *new_value, int modify_type, int stage, int force_change) /* {{{ */
        2. {
        3. zend_ini_entry *ini_entry;
        4. zend_string *duplicate;
        5. zend_bool modifiable;
        6. zend_bool modified;
        7. //EG(modified_ini_directives)用于存放被修改过的ini_entry,根据name(配置名称)寻找到对应ini_entry
        8. if ((ini_entry = zend_hash_find_ptr(EG(ini_directives), name)) == NULL) {
        9. return FAILURE;
        10. }
        11. modifiable = ini_entry->modifiable;
        12. modified = ini_entry->modified;
        13. if (stage == ZEND_INI_STAGE_ACTIVATE && modify_type == ZEND_INI_SYSTEM) {
        14. ini_entry->modifiable = ZEND_INI_SYSTEM;
        15. }
        16. if (!force_change) {
        17. if (!(ini_entry->modifiable & modify_type)) {
        18. return FAILURE;
        19. }
        20. }
        21. if (!EG(modified_ini_directives)) {
        22. ALLOC_HASHTABLE(EG(modified_ini_directives));
        23. zend_hash_init(EG(modified_ini_directives), 8, NULL, NULL, 0);
        24. }
        25. //不管我们先后在php代码中调用几次ini_set,只有第一次ini_set时才会进入这段逻辑,设置orig_value。从第二次调用ini_set开始,便不会再次执行这段分支,因为此时的modified已经被置为1了。因此,ini_entry->orig_value始终保存的是第一次修改之前的配置值(即最原始的配置)
        26. if (!modified) {
        27. ini_entry->orig_value = ini_entry->value;
        28. ini_entry->orig_modifiable = modifiable;
        29. ini_entry->modified = 1;
        30. zend_hash_add_ptr(EG(modified_ini_directives), ini_entry->name, ini_entry);
        31. }
        32. duplicate = zend_string_copy(new_value);
        33. //调用on_modify是为了能够更新模块的全局变量。每一个ini_entry中都存储了该模块全局变量的地址以及对应的偏移量,使得on_modify可以很迅速的进行内存修改。
        34. if (!ini_entry->on_modify
        35. || ini_entry->on_modify(ini_entry, duplicate, ini_entry->mh_arg1, ini_entry->mh_arg2, ini_entry->mh_arg3, stage) == SUCCESS) {
        36. if (modified && ini_entry->orig_value != ini_entry->value) { /* we already changed the value, free the changed value */
        37. zend_string_release(ini_entry->value);
        38. }
        39. ini_entry->value = duplicate;
        40. } else {
        41. zend_string_release(duplicate);
        42. return FAILURE;
        43. }
        44. return SUCCESS;
        45. }
      19. 可以看出该函数是同过通过 on_modify 回调函数直接修改了内存中的全局变量而达到控制执行时间的目的,所以这也解释了为什么ini_set 在执行结束就会失效。先说这些,过几天我会整理一下把整个PHP生命周期的ini加载过程详细总结一下。

      20. 参考文章:http://www.cnblogs.com/driftc...

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/28761.html

相关文章

  • php的异常和处理

    摘要:常见错误处理类型语法错误环境错误逻辑错误常见错误级别最低级别的错误不推荐,不建议,使用一些过期函数的时候会出现,程序继续执行通知级别的错误使用一些未定义变量常量或者数组没有加引号的时候会出现,程序继续执行警告级别的错误程序出问题了,需要修改 常见错误处理类型 语法错误 环境错误 逻辑错误 常见错误级别 Deprecated 最低级别的错误 不推荐,不建议,使用一些过期函数的时候...

    CarterLi 评论0 收藏0
  • php error_reporting()关闭报错

    摘要:至,有同样的行为。表示关闭所有错误报告表示显示二函数说明设置应该报告何种错误说明函数能够在运行时设置指令。后果是导致脚本终止不再继续运行。初始化启动过程中发生的警告非致命错误。用户产少的警告信息。出外的所有错误和警告信息。 错误报告级别:指定了在什么情况下,脚本代码中的错误(这里的错误是广义的错误,包括E_NOTICE注意、E_WARNING警告、E_ERROR致命错误等)会以错误报告...

    noONE 评论0 收藏0
  • Xdebug中文文档-堆栈跟踪

    摘要:英文原始文档地址中文文档地址当被激活时,只要决定显示通知,警告,错误等,就会显示堆栈跟踪。堆栈跟踪中的变量默认情况下,将在它生成的堆栈跟踪中显示可变信息。 文档内容来自xdebug.org/docs,翻译时xdebug版本为2.6。我在官方文档基础上针对中文排版和教程内容的编排做了一些优化,希望中文文档看起来更容易理解。 英文原始文档地址:https://xdebug.org/docs...

    wzyplus 评论0 收藏0
  • PHP 实现定时任务的几种方法

    摘要:为系统增加的第一行代码不会影响该脚本在下的运行,因此您也可以用该方法编写跨平台的脚本程序。指定会话页面在客户端中的有限期分钟缺省下为分钟。最原始的博主没有找到,只能在此声明,特为转载。 这几天需要用PHP写一个定时抓取网页的服务器应用. 在网上搜了一下解决办法, 发现OSchina的 一个问题的解答很精彩(值得一看,谢谢大牛们的精彩回答O(∩_∩)O~), 提出几种解决办法.现总结如下...

    huhud 评论0 收藏0
  • 开发模式与产品模式下的PHP报错处理

    摘要:当程序开发完成,成为正式产品时,我们希望将没有预测到的报错信息记录到错误日志中,而不是将这些报错信息展示给用户,因为用户极有可能利用这些暴露出脚本路径数据库信息或其他的报错信息进行一些破坏性的黑客行动。 程序报错总是在所难免,尽管我们书写代码时已经格外小心。 在开发php程序时,我们希望遇到php报错,可以第一时间展示给我们,以便于调试。当程序开发完成,成为正式产品时,我们希望将没有预...

    baiy 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<