本文转自http://blog.csdn.net/cedricliang/article/details/17247749?9435;这是在我想在js的循环中加入一段php,这段php代码会在每次执行时引用一个全局变量并++,但是我发现这个全局变量每次输出的结果相同,没有想象中的++,因此学习了一下php的生命周期,转文详细说明了php的生命周期期,特此转载以备忘。

首先开始介绍php的生命周期,了解一个php程序从开始运行到最后结束究竟经过怎么样的过程,对学习php和平时php开发应该是很重要的。

起始和关闭阶段:

对于php的起始和关闭阶段可以分成两层,
第一层是php解释器作为一个整体进行结构和值的初始化过程。
第二层则是在每一个页面的请求过程中。
对于每个扩展而言,都会有一个初始化MINT函数,这个过程会声明变量、类,注册资源、流和过滤处理器,这些操作在所有的请求中都是存在的,所以可以称为是Persistent的。一般进行如下的两步操作:
REGISTER_INI_ENTRIES()
初始化模块的全局变量
而在页面发出请求了之后,PHP则会建立一个包括符号表和配置值在内的操作环境,然后这次php解释器会再次循环每一个扩展,调用每个扩展的RINIT初始化函数。一般RINT函数的操作如下:
把globalvalue设成默认值,这些全局变量往往是每个请求都需要的,但是针对每一个请求而言都是相互独立的。
那些需要用到的变量需要放到符号表中以备调用。
在这个函数中还可以记录一下请求的相关信息
完成了请求的处理之后,如果到达了脚本尾部或通过die() exit()推出了,那么php通过调用RSHUTDOWN()开始清理
这时候每个符号表中的变量都会被unset掉
而当所有的请求都被满足了之后,就开始针对于模块的MSHUTDOWN过程
调用UNREGISTER_INI_ENTRYES(),与MINIT函数的初始化过程相互对应。
php程序执行的生命周期:
要了解生命周期,就必须对不同的执行方式有所涉及。php可以有几种不同的执行方式,每种方式都有其特定的生命周期。 CLI:这个是从命令行执行php程序,它的生命周期最为简单。比如在执行test.php的时候,就经历了如下的过程 图1 CLI进行php执行的过程
注意到MINIT RINIT RSHUTDOWN MSHUTDOWN都只被调用过一次,这个比较类似于瀑布式的结构。
多线程的模块方式:这是最常用的一种方式,php作为APXS的模块对apache进行配合。在apache启动的时候会fork很多的子进程。针对于多个不同的请求,配合的是多个不同的初始化与结束过程。但是对每一个线程来说,只有一个MINIT和MSHUTDOWN的调用。而每个请求都对应着自己单独的RINIT和RSHUTDOWN。
多线程的模块方式:采用多线程的方法可以避免不同的线程重复的调用MINIT/MSHUTDOWN.它具有的好处是多个请求可以共享信息,但是请求之间的隔离要求比较高,不然容易出现变量的访问出错。 Zend 线程安全
php对线程安全的处理有专门的机制,称为Thread Safe Resource Management(TSRM)。在进行线程安全和非线程安全声明的时候,明显有一些不同之处: 线程安全的变量声明:
typedef struct {
int sampleint;
char *samplestring;
} php_sample_globals;
int sample_globals_id;
PHP_MINIT_FUNCTION(sample)
{
ts_allocate_id(&sample_globals_id,
sizeof(php_sample_globals),
(ts_allocate_ctor) php_sample_globals_ctor,
(ts_allocate_dtor) php_sample_globals_dtor);
return SUCCESS;
}
从这段代码中可以看到,在MINIT阶段,需要通过ts_allocate_id函数来通知TSRM这个程序需要多少空间,TSRM会增加当前的空间消耗,并返回一个id指向线程数据池的相应部分。
而当请求要访问数据的时候,就首先从TSRM层找到当前线程的资源池的指针,然后加上ts_allocate_id()返回的资源id作为offset。
非线程安全的变量声明:
typedef struct {
int sampleint;
char *samplestring;
} php_sample_globals;
php_sample_globals sample_globals;
PHP_MINIT_FUNCTION(sample)
{
php_sample_globals_ctor(&sample_globals TSRMLS_CC);
return SUCCESS;
}
在非线程安全的情况下,只需要简单的声明变量,更加快速有效,数据的地址在编译阶段就可以确定,而不是像在线程安全的情况下需要运行时计算。同时它还有一个好处就是一段程序出了bug不会让整个webserver坏掉。因此在不需要用到线程安全的时候,还是应该尽量使用非线程安全声明方式。
为了构建线程安全的php,必须在编译的时候加上--enable-maintainer-zts选项。而在程序中检测的时候可以用#ifdef ZTS,通过这种检测可以实现对全局变量声明的不同处理,入下例所示:
#ifdef ZTS
#define HELLO_G(v) TSRMG(hello_globals_id, zend_hello_globals *, v)
#else
#define HELLO_G(v) (hello_globals.v)
#endif 通过判断是否启动了线程安全,对变量进行了不同的声明和访问方式指定。
在启用了线程安全之后,为了在不同的线程之间对数据进行区分,PHP会自动启用一个tsrm_ls指针。在这个指针的帮助下,各个线程的函数才能找到对应的符号表进行相应的变量读取操作。也正是这种机制使得多个线程一起工作的时候,数据空间不会乱掉。 通过对php的起始和结束阶段、生命周期和Zend线程安全的机制的了解,有利于后续对php扩展编译的研究。

(转)php的扩展和嵌入--php的生命周期与变量详述的更多相关文章

  1. Python 7 —— 扩展与嵌入

    Python 7 —— 扩展与嵌入 所谓扩展是指,在Python当中调用其他语言,由于Python的问题主要是效率,这里的扩展主要是指扩展C C++程序(重点) 所谓嵌入是指,在其他语言当中可以调用P ...

  2. 《扩展和嵌入python解释器》1.4 模块方法表和初始化函数

    <扩展和嵌入python解释器>1.4 模块方法表和初始化函数   1.4 模块方法表和初始化函数 下面,我演示如何从Python程序调用spam_system().首先,我们需要在’方法 ...

  3. PHP扩展-生命周期和内存管理

    1. PHP源码结构 PHP的内核子系统有两个,ZE(Zend Engine)和PHP Core.ZE负责将PHP脚本解析成机器码(也成为token符)后,在进程空间执行这些机器码:ZE还负责内存管理 ...

  4. 对微信小程序的生命周期进行扩展 – Typescript 篇

    最近利用业余时间倒腾了一个微信小程序,主要目的是横向比较一些业界小程序平台的架构和做法.因为有在其他平台长期的开发经验,对于小程序的一些机制做了一些辩证的思考.例如,小程序的页面,其实不是一个页面,而 ...

  5. Spring Boot中的那些生命周期和其中的可扩展点(转)

    前言可扩展点的种类Spring Boot启动过程 1.SpringApplication的启动过程 2.ApplicationContext的启动过程 3.一般的非懒加载单例Bean在Spring B ...

  6. 扩展和嵌入 Python 解释器 用 C 或 C++ 编写模块以使用新模块来扩展 Python 解释器的功能 定义新的函数\对象类型\方法。 将 Python 解释器嵌入到另一个应用程序中

    // https://python3-cookbook.readthedocs.io/zh_CN/latest/c15/p02_write_simple_c_extension_module.html ...

  7. c++对象在lua层的生命周期与内容扩展

    前言 上一篇博客记录了 tolua++ 将 c++类型,变量,函数,以及对象导出到 lua 的过程,这篇博客就接着记录一下 c++对象的内存回收以及c++对象数据和方法在lua中的扩展. 首先 tol ...

  8. JavaScript基本语法(JavaScript代码嵌入方式与声明和使用变量)

    .JavaScript代码嵌入方式 #①HTML文档内 JavaScript代码要写在script标签内 script标签可以写在文档内的任意位置 为了能够方便查询或操作HTML标签(元素)scrip ...

  9. iOS开发系列--App扩展开发

    概述 从iOS 8 开始Apple引入了扩展(Extension)用于增强系统应用服务和应用之间的交互.它的出现让自定义键盘.系统分享集成等这些依靠系统服务的开发变成了可能.WWDC 2016上众多更 ...

随机推荐

  1. Support Library官方教程(1)概述

    Support Library The Android Support Library package is a set of code libraries that provide backward ...

  2. BZOJ 2339 卡农(组合数学)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2339 题意: 思路: i64 Pow(i64 a,i64 b,i64 mod){    ...

  3. java.lang.NoSuchMethodError: No static method setLayoutDirection(Landroid/graphics/drawable/Drawable;I)V in class Landroid/support/v4/graphics/drawable/DrawableCompat

    Bug: java.lang.NoSuchMethodError: No static method setLayoutDirection(Landroid/graphics/drawable/Dra ...

  4. 自定义sublime代码片段

    sublime text 已经有一些他们内置的一些代码片段,但是有时候,这些并不能满足我们,这就需要我们自定义一些代码片段. 步骤如下: 1.打开sublime text 2.选择 tools -&g ...

  5. ZOJ 3607 Lazier Salesgirl(贪心)

    题目:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3607 题意:一个卖面包的小姑娘,给第i个来买面包的人的价格是pi, ...

  6. 如何使java中double类型不以科学计数法表示

    在java中,把一个double或者BigDecimal的小数转换为字符串时,经常会用科学计数法表示,而我们一般不想使用科学计数法,可以通过:DecimalFormat a = new Decimal ...

  7. ArrayList和List之间的转换

    开发中不免碰到List与数组类型之间的相互转换,举一个简单的例子: package test.test1; import java.util.ArrayList; import java.util.L ...

  8. poj3683 Priest John's Busiest Day

    2-SAT. 读入用了黄学长的快速读入,在此膜拜感谢. 把每对时间当作俩个点.如果有交叉代表相互矛盾. 然后tarjan缩点,这样就能得出当前的2-SAT问题是否有解. 如果有解,跑拓扑排序就能找出一 ...

  9. Qt之自定义界面(窗体缩放-跨平台终极版)

    简述 通过上一节内容,我们实现了窗体的缩放,功能很不错,但是很遗憾-不支持跨平台!如果对于多平台来说,这是一个硬伤,所以,我们急需要一个能够支持跨平台的实现方案. 在网上看到过很多不同的实现方式,多多 ...

  10. UVa 1149 Bin Packing 【贪心】

    题意:给定n个物品的重量l[i],背包的容量为w,同时要求每个背包最多装两个物品,求至少要多少个背包才能装下所有的物品 和之前做的独木舟上的旅行一样,注意一下格式就好了 #include<ios ...