对于PHP的中的数据来源, 不外乎有俩种:

1. 来自代码中

2. 来自外部(GET/POST/DB)

对于代码中的变量(也就是直接量)来说, 变量分配/赋值在编译期, 活跃在执行器, 在请求关闭期被销毁.对于这些变量来说, 使用APC进行Opcode缓存, 则会缓存这部分变量的值.

而对于来自外部的变量, 变量分配/赋值在编译器后, 执行期前, 在请求关闭期被销毁,对于这些变量来说, 使用APC进行OpCode缓存, 是不会被缓存的.

今天就着重关注下外部变量的一个部分,GET来的数据的整个生命周期.

假设, 有如下请求到来:

GET /index.php?name=laruence&career[]=yahoo&career[]=baidu

而, 在index.php中:

$name = $_GET['name'];

$career = $_GET['career']; //array

我们知道, 在最后的执行期, $_GET数组必然包含如下片段:

$_GET = array(

'name' => 'laruence',

'career' => array(

'yahoo', 'baidu',

),

)

那么, 我们今天就重点关注下, Query String是如何构建成_GET数组的(关于GET变量的生成, 请一并阅读我之前的文章: "PHP的GET/POST等大变量生成过程"):

在请求到来时刻,php_request_startup(定义在main.c)被调用,来做初始化现场. 在这个过程中包括设置超时值,调用各个模块的请求初始化函数. 当然也包括我们关心的, 创建变量环境.

php_hash_environment根据php.ini中的variables_order来依次初始化各个预定义大变量, 那么对于$_GET来说:

...

case 'g':

case 'G':

if (!_gpc_flags[2]) {

sapi_module.treat_data(PARSE_GET, NULL, NULL TSRMLS_CC);

_gpc_flags[2] = 1;

if (PG(register_globals)) {

php_autoglobal_merge(&EG(symbol_table),

Z_ARRVAL_P(PG(http_globals)[TRACK_VARS_GET]) TSRMLS_CC);

}

}

break;

...

大体可以看出,这段逻辑,首先通过treat_data来生成变量hash(PG(http_globals)[TRACK_VARS_GET]), 如果打开了auto_register_globals,则再把$_GET数组中的变量加入到符号表中.

treat_data是属于sapi_module_struct中的一个成员:

注意:本文基于apache2handler方式的sapi, 这个启动过程和之前的文章sapi

原理中的启动过程略有不同, php5通过注册apache2的ap_hook_post_config挂钩,

在apache server启动的时候启动php(php_apache_server_startup,定义在

sapi/apache2hander/sapi_apache2.c中), 在这个函数中调用sapi_startup启动sapi,

继而通过调用php_apache2_startup来注册sapi module struct,

然后调用php_module_startup来初始化PHP, 其中又会初始化ZEND引擎,

以及填充zend_module_struct中的treat_data成员(通过php_startup_sapi_content_types)

为php_default_treat_data

现在回过头来继续看treat_data(也就是php_default_treat_data):

....

if (arg == PARSE_GET) { /* GET data */

c_var = SG(request_info).query_string;

if (c_var && *c_var) {

res = (char *) estrdup(c_var);

free_buffer = 1;

} else {

free_buffer = 0;

}

} else if (arg == PARSE_COOKIE) { /* Cookie data */

....

在上面的逻辑中, 给res复制为query_string, SG(request_info)是一个代表了当前请求信息的结构体, 其中query_string是在php_apache_request_ctor中通过复制apache的reqeust_rec结构体中的args而来的.

对于本文的例子来说, 此时res即为"name=laruence&career[]=yahoo&career[]=baidu",

继续在treat_data中, 随后的逻辑是:

var = php_strtok_r(res, separator, &strtok_buf);

...

while (var) {

val = strchr(var, '=');

if (arg == PARSE_COOKIE) {

/* Remove leading spaces from cookie names,

needed for multi-cookie header where ; can be followed by a space */

while (isspace(*var)) {

var++;

}

if (var == val || *var == '\0') {

goto next_cookie;

}

}

if (val) { /* have a value */

int val_len;

unsigned int new_val_len;

*val++ = '\0';

php_url_decode(var, strlen(var));

val_len = php_url_decode(val, strlen(val));

val = estrndup(val, val_len);

if (sapi_module.input_filter(arg, var, &val, val_len, &new_val_len TSRMLS_CC)) {

php_register_variable_safe(var, val, new_val_len, array_ptr TSRMLS_CC);

}

efree(val);

} else {

...

首先, 通过php_strtok_r把res根据"&"分割成一个一个的"key=value"段, 接下来分别为var和val复制为key和value, 注意到这个过程中会分别对var和val做php_url_decode.

最后通过php_register_variable_safe, 给array_ptr(此时指向PG(http_globals)[TRACK_VARS_GET], 也就是$_GET)添加一个名为var值为val的成员.

到了这一步, 我们的$_GET数组中, 就包含了如下的成员:

'name' => 'laruence',

'career' => array(

'yahoo', 'baidu',

),

未完待续(变量的销毁过程)...

php 生命变量,深入理解PHP原理之变量生命期(一)相关推荐

  1. 深入理解PHP原理之变量分离/引用(Variables Separation)

    引自: http://www.laruence.com/ [风雪之隅 ] 在前面的文章中我已经介绍了PHP的变量的内部表示(深入理解PHP原理之变量(Variables inside PHP)),以及 ...

  2. 深入理解PHP原理之变量作用域

    作者:laruence(http://www.laruence.com/) 地址: http://www.laruence.com/2008/08/26/463.html                ...

  3. 深入理解PHP原理之变量(Variables inside PHP)

    或许你知道,或许你不知道,PHP是一个弱类型,动态的脚本语言.所谓弱类型,就是说PHP并不严格验证变量类型(严格来讲,PHP是一个中强类型语言,这部分内容会在以后的文章中叙述),在申明一个变量的时候, ...

  4. python的变量如何理解_Python程序中变量作用范围应该如何理解?

    初学Python,自己尝试着写了一个爬虫,主要代码如下import json import scrapy from bs4 import BeautifulSoup from docx import ...

  5. C++ 私有成员变量的理解

    私有成员变量的概念,在脑海中的现象是,以private关键字声明,是类的实现部分,不对外公开,不能在对象外部访问对象的私有成员变量. 然而,在实现拷贝构造函数和赋值符函数时,在函数里利用对象直接访问了 ...

  6. 【数据收集】名义变量、序级变量、区间变量、比率变量的理解及例子

    [数据收集]名义变量.序级变量.区间变量.比率变量的理解及例子 名义变量(Nominal Variable) 分类及举例 序级变量(Ordinal Variable) 分类和举例 区间变量(Inter ...

  7. 《深入理解mybatis原理》 MyBatis缓存机制的设计与实现

    本文主要讲解MyBatis非常棒的缓存机制的设计原理,给读者们介绍一下MyBatis的缓存机制的轮廓,然后会分别针对缓存机制中的方方面面展开讨论. MyBatis将数据缓存设计成两级结构,分为一级缓存 ...

  8. [笔试题]交换两个数不使用第三方变量 深入理解按位异或运算符

    异或运算相当与mod 2运算: 1^1 = 0, 1^0 = 1, 0^1= 1, 0 ^ 0 = 0 (1+1)%2 = 0, (1+0)%2 = 1, (0+1)%2 = 1, (0+0)%2 = ...

  9. 深入理解mybatis原理, Mybatis初始化SqlSessionFactory机制详解(转)

    文章转自http://blog.csdn.net/l454822901/article/details/51829785 对于任何框架而言,在使用前都要进行一系列的初始化,MyBatis也不例外.本章 ...

最新文章

  1. Huggingface及BERT代码介绍
  2. 后盾网lavarel视频项目---vue实现动态添加和删除板块
  3. 如何成为一名大数据工程师?
  4. 多线程编程指南 part 2
  5. 谈判失败:Oracle杀死Java EE
  6. 特征工程系列学习(零)引言
  7. aes加密php源码,AES加解密类源码 · ThinkPHP5高阶实战教程 --诠释为API开发而生 · 看云...
  8. 《微服务设计》(一)---- 微服务
  9. Java常见面试题:对象的访问定位的两种方式
  10. C/C++(变量作用域)
  11. 乌卡时代下,企业供应链管理体系的应对策略
  12. 皮卡智能联手全球最大贸易服务商PingPong,共推AIGC应用落地服务
  13. css渐变描边视频,CSS制作渐变描边等文字特效
  14. 快递100API 手机H5使用
  15. crossorigin
  16. LK金字塔光流法与简单实现
  17. STM32操作24位AD芯片ADS1246
  18. 众筹,帮创业者跨过“死亡之谷”
  19. java使用jxls导出excel功能
  20. 彼得·德鲁克--管理思想摘录

热门文章

  1. Java一个简单的爬虫:爬去网页代码
  2. 外链引入css有哪些方式_CSS 文件的4种引入方式
  3. javaWeb三大框架总结
  4. python查看对象占用内存_『Python』内存分析_List对象内存占用分析
  5. java 多线程 选择题_Java多线程之三道多线程练习题
  6. git rollback代码都没了_ECBM库也能自动更新吗?——论GIT的用法
  7. ntp协议原理linux网络编程,NTP协议
  8. 服务器u盘安装win7系统,如何用U盘安装win7原版64位系统
  9. linux启用shell脚本,linux下开机启动shell脚本
  10. OpenShift 4.3 通过脚本配置ServiceMesh和Serverless运行环境