热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

探索PHP生命周期

PHP的生命周期是一个很复杂的过程,其生命周期应该被热衷于使用它的人所掌握,主要内容如下,PHP启动,如果运行的是CLI或者FPM,它将运行【C main()】。

在你分析 PHP 扩展结构时(zend_module_entry 结构),这些钩子可以清晰地看到:

struct _zend_module_entry {
        unsigned short size;
        unsigned int zend_api;
        unsigned char zend_debug;
        unsigned char zts;
        const struct _zend_ini_entry *ini_entry;
        const struct _zend_module_dep *deps;
        const char *name;
        const struct _zend_function_entry *functions;
        int (*module_startup_func)(INIT_FUNC_ARGS);        /* MINIT() */
        int (*module_shutdown_func)(SHUTDOWN_FUNC_ARGS);   /* MSHUTDOWN() */
        int (*request_startup_func)(INIT_FUNC_ARGS);       /* RINIT() */
        int (*request_shutdown_func)(SHUTDOWN_FUNC_ARGS);  /* RSHUTDOWN() */
        void (*info_func)(ZEND_MODULE_INFO_FUNC_ARGS);     /* PHPINFO() */
        const char *version;
        size_t globals_size;
#ifdef ZTS
        ts_rsrc_id* globals_id_ptr;
#else
        void* globals_ptr;
#endif
        void (*globals_ctor)(void *global);                /* GINIT() */
        void (*globals_dtor)(void *global);                /* GSHUTDOWN */
        int (*post_deactivate_func)(void);                 /* PRSHUTDOWN() */
        int module_started;
        unsigned char type;
        void *handle;
        int module_number;
        const char *build_id;
};

现在让我们看看你应该在这些钩子中编写哪种代码。

模块初始化:MINIT()

这是 PHP 进程启动步骤。在扩展的MINIT()中,你将加载并分配以后每次请求需要的任何持久对象或者信息。它们的大部分将分配为只读对象。

MINIT()中,尚未有线程或进程弹出,所以你完全可以访问全局变量,而没有任何保护。另外,由于请求尚未启动,因此你不能分配请求绑定的内存。你永远不会在MINIT()步骤中使用Zend 内存管理 分配,但是会使用永久分配。不是 emalloc(),而是pemalloc()。否则会导致崩溃。

MINIT()中,执行引擎仍未启动,所以不要在没有特别注意的情况下,尝试访问其任何结构。

如果你需要为你的扩展注册 INI 入口,则MINIT() 是正确的做法。

如果你要为以后使用而注册只读zend_strings,请使用持久分配了。

如果你需要分配的对象在处理请求时会写入,那么你必须复制它们的内存分配到该请求的线程专用池。记住,你只可以在MINIT()中安全地写入全局空间。

注意

内存管理、分配和调试是内存管理的部分章节。

在 php_module_startup()函数中,通过zend_startup_modules()触发 MINIT()

模块终止:MSHUTDOWN()

这是 PHP 进程终止步骤。很容易, 基本上,你在这里运行了与MINIT()中使用的相反的操作。你释放了资源,取消 INI 设置的注册等等。

再次注意:执行引擎是关闭的,所以你不应在此处访问其任何变量。

由于你在此处不需要请求,所以不应使用Zend 内存管理 的efree() 或类似函数去释放资源,但对于释放持久分配,使用pefree()

在php_module_shutdown()函数中,由zend_shutdown()zend_destroy_modules()中触发MSHUTDOWN()

请求初始化: RINIT()

刚刚看过的请求,PHP 将在这里处理它。在RINIT()中,你引导了处理该精确请求所需的资源。PHP 是一种无共享架构,它提供了内存管理功能。

RINIT()中,如果需要分配动态内存,你将使用Zend 内存管理器。你将调用 emalloc()。Zend 内存管理器 追踪你通过它分配的内存,当请求关闭时,如果你忘记这么做,它将尝试释放请求绑定的内存(你不应这么做)。

在这里,你不应请求持久的动态内存,即 libc 的malloc() 或Zend 的pemalloc()。如果你在这里请求持久内存,并且忘记释放它,则将造成泄露,并且随着 PHP 处理越来越多的请求而堆积,最终导致进程崩溃(Kernel OOM) ,并且导致机器内存不足。

另外,务必注意不要在这里写入全局空间。如果 PHP 作为选定的并行模型运行到线程中,那么你将修改每个线程池中的上下文(所有与你的请求并行处理的请求),并且如果你没有锁定内存,也可能触发竞争条件。如果你需要全局,你必须保护它们。

注意

全局范围管理解释在专用章节。

在php_request_startup()函数中,通过zend_activate_module()触发RINIT()

请求终止: RSHUTDOWN()

这是 PHP 请求终止步骤。PHP 刚结束处理其请求,现在来清理其部分作为无共享架构的内存。接下来的请求不应记住当前请求的任何内容。很容易,基本上,你在此处执行了与RINIT()使用的相反的操作。你释放了请求绑定的资源。

由于你在此处使用了请求,你应使用 Zend 内存管理器的efree()或类似方式释放资源。如果你忘记释放并且造成泄露,在调试版本下,内存管理器将在进程stderr上记录关于泄露的指针的日记,并且将为你释放它们。

给你个主意,RSHUTDOWN()将被调用:

  • 执行用户区关闭功能后 (register_shutdown_function())
  • 在调用每个对象析构函数之后
  • PHP 输出缓冲区刷新之后
  • 禁用 max_execution_time 之后

在php_request_shutdown()函数中,通过zend_deactivate_modules()触发RSHUTDOWN()

Post 请求终止: PRSHUTDOWN()

这个钩子很少使用。它在 RSHUTDOWN()之后调用,但是中间还会运行一些额外的引擎代码。
尤其是在 Post-RSHUTDOWN 中:

  • PHP 输出缓冲区已关闭,并且它的处理程序已刷新
  • PHP 超全局已经销毁
  • 执行引擎已经关闭

这个钩子很少使用。在php_request_shutdown()函数中,通过zend_post_deactivate_modules(),在RSHUTDOWN()之后被触发。

全局初始化: GINIT()

线程库每次弹出线程时都会调用该钩子。如果你使用多进程,当 PHP 启动,仅在触发 MINIT() 之前调用此函数。

这里不讲太多细节,只需在这里简单地初始化全局变量,通常初始化为0。全局管理将在专用章节详细说明。

记住,全局变量不会在每次请求后清理。如果你需要为每次新的请求重置它们(可能),那么你必须将这样地进程放到RINIT()中。

注意

全局范围管理在专用章节详细介绍。

全局终止: GSHUTDOWN()

在线程库中,每当线程终止时都会调用该钩子。如果你使用多线程,该函数将在 PHP 终止期间(在MSHUTDOWN())被调用一次。

在这里不提供太多细节,你只需简单地在这里取消初始化你的全局变量,通常你不必做什么,但如果在构建全局(GINIT())时分配了资源,在这里的步骤你应该释放它们。

全局管理将在专用章节详细介绍。

记住,全局变量在每次请求后不会清除。即GSHUTDOWN()不会作为RSHUTDOWN()的一部分被调用。

注意

全局范围管理在专用章节有详细介绍。

信息收集: MINFO()

该钩子很特殊,它永远不会被引擎自动触发,只有你询问它有关扩展的信息时才会触发。典型的例子是调用phpinfo()。然后运行此函数,并将有关当前扩展的特殊信息打印到流中。

简而言之,phpinfo() 展示信息。

该函数也可以通过 CLI 使用反射开关之一调用,例如php --ri pib 或通过用户区调用ini_get_all()

你可以将其留空,在这种情况下,只有扩展的名字显示,没有其他(可能不会显示 INI 设置,因为这是 MINFO() 的一部分)。

关于 PHP 生命周期的思考

你可能已经发现了,RINIT()RSHUTDOWN() 尤其重要,因为它们在扩展中被触发成千上万次。如果 PHP 步骤是关于 Web (不是 CLI),并且已经配置为可以处理无数次请求,那么你的 RINIT()/RSHUTDOWN() 组将被无数次调用。

我们想要再次引起你对内存管理的关注。在处理请求时(在RINIT()RSHUTDOWN()之间),你最终泄露的小字节,将对满载服务器产生严重影响。这就是为什么建议你使用 Zend 内存管理器 进行此类分配,并且准备好调试内存布局。作为无共享架构的一部分,PHP 在每次请求最后都会忘记并释放请求内存,这是 PHP 的内部设计。

另外,如果你的崩溃信号是 SIGSEGV (坏内存访问),则整个进程会崩溃。如果 PHP 是使用线程作为多进程引擎,那么你所有其他线程也将崩溃,甚至可能造成服务器崩溃。

注意

C 语言不是 PHP 语言。使用 C,在程序的错误很可能导致程序的崩溃与终止。

通过重写函数指针进行挂钩

现在你知道引擎何时会触发代码,还存在值得注意的函数指针,你可以替换它们来挂载到引擎。因为那些指针是全局变量,因此你可以将它们替换为 MINIT() 步骤,并将它们放回MSHUTDOWN()中。

感兴趣的有:

  • AST, Zend/zend_ast.h:

    • void (zend_ast_process_t)(zend_ast ast)
  • Compiler, Zend/zend_compile.h:

    • zend_op_array (zend_compile_file)(zend_file_handle file_handle, int type)*
    • zend_op_array (zend_compile_string)(zval source_string, char filename)
  • Executor, Zend/zend_execute.h:

    • void (zend_execute_ex)(zend_execute_data execute_data)
    • void (zend_execute_internal)(zend_execute_data execute_data, zval return_value)*
  • GC, Zend/zend_gc.h:

    • int (gc_collect_cycles)(void)*
  • TSRM, TSRM/TSRM.h:

    • void (tsrm_thread_begin_func_t)(THREAD_T thread_id)*
    • void (tsrm_thread_end_func_t)(THREAD_T thread_id)*
  • Error, Zend/zend.h:

    • void (zend_error_cb)(int type, const char error_filename, const uint error_lineno, const char format, va_list args)*
  • Exceptions, Zend/zend_exceptions.h:

    • void (zend_throw_exception_hook)(zval ex)
  • Lifetime, Zend/zend.h:

    • void (zend_on_timeout)(int seconds)*
    • void (zend_interrupt_function)(zend_execute_data execute_data)
    • void (zend_ticks_function)(int ticks)*

还有其他存在,但是上面的是最重要的,当你设计 PHP 扩展时,你可能需要。因为它们的名字很容易看,所以不再详细解释它们。

如果你需要更多信息,你可以在 PHP 源代码查看,并发现何时和如何触发它们。

以上就是探索PHP 生命周期的详细内容,更多请关注 第一PHP社区 其它相关文章!


推荐阅读
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 如何实现织梦DedeCms全站伪静态
    本文介绍了如何通过修改织梦DedeCms源代码来实现全站伪静态,以提高管理和SEO效果。全站伪静态可以避免重复URL的问题,同时通过使用mod_rewrite伪静态模块和.htaccess正则表达式,可以更好地适应搜索引擎的需求。文章还提到了一些相关的技术和工具,如Ubuntu、qt编程、tomcat端口、爬虫、php request根目录等。 ... [详细]
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 搭建Windows Server 2012 R2 IIS8.5+PHP(FastCGI)+MySQL环境的详细步骤
    本文详细介绍了搭建Windows Server 2012 R2 IIS8.5+PHP(FastCGI)+MySQL环境的步骤,包括环境说明、相关软件下载的地址以及所需的插件下载地址。 ... [详细]
  • 这是原文链接:sendingformdata许多情况下,我们使用表单发送数据到服务器。服务器处理数据并返回响应给用户。这看起来很简单,但是 ... [详细]
  • PHP设置MySQL字符集的方法及使用mysqli_set_charset函数
    本文介绍了PHP设置MySQL字符集的方法,详细介绍了使用mysqli_set_charset函数来规定与数据库服务器进行数据传送时要使用的字符集。通过示例代码演示了如何设置默认客户端字符集。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • Centos7.6安装Gitlab教程及注意事项
    本文介绍了在Centos7.6系统下安装Gitlab的详细教程,并提供了一些注意事项。教程包括查看系统版本、安装必要的软件包、配置防火墙等步骤。同时,还强调了使用阿里云服务器时的特殊配置需求,以及建议至少4GB的可用RAM来运行GitLab。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • 禁止程序接收鼠标事件的工具_VNC Viewer for Mac(远程桌面工具)免费版
    VNCViewerforMac是一款运行在Mac平台上的远程桌面工具,vncviewermac版可以帮助您使用Mac的键盘和鼠标来控制远程计算机,操作简 ... [详细]
  • 本文详细介绍了云服务器API接口的概念和作用,以及如何使用API接口管理云上资源和开发应用程序。通过创建实例API、调整实例配置API、关闭实例API和退还实例API等功能,可以实现云服务器的创建、配置修改和销毁等操作。对于想要学习云服务器API接口的人来说,本文提供了详细的入门指南和使用方法。如果想进一步了解相关知识或阅读更多相关文章,请关注编程笔记行业资讯频道。 ... [详细]
  • 本文介绍了使用PHP实现断点续传乱序合并文件的方法和源码。由于网络原因,文件需要分割成多个部分发送,因此无法按顺序接收。文章中提供了merge2.php的源码,通过使用shuffle函数打乱文件读取顺序,实现了乱序合并文件的功能。同时,还介绍了filesize、glob、unlink、fopen等相关函数的使用。阅读本文可以了解如何使用PHP实现断点续传乱序合并文件的具体步骤。 ... [详细]
author-avatar
mobiledu2502889521
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有