php 生命变量,深入理解PHP原理之变量生命期(一)
对于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原理之变量生命期(一)相关推荐
- 深入理解PHP原理之变量分离/引用(Variables Separation)
引自: http://www.laruence.com/ [风雪之隅 ] 在前面的文章中我已经介绍了PHP的变量的内部表示(深入理解PHP原理之变量(Variables inside PHP)),以及 ...
- 深入理解PHP原理之变量作用域
作者:laruence(http://www.laruence.com/) 地址: http://www.laruence.com/2008/08/26/463.html ...
- 深入理解PHP原理之变量(Variables inside PHP)
或许你知道,或许你不知道,PHP是一个弱类型,动态的脚本语言.所谓弱类型,就是说PHP并不严格验证变量类型(严格来讲,PHP是一个中强类型语言,这部分内容会在以后的文章中叙述),在申明一个变量的时候, ...
- python的变量如何理解_Python程序中变量作用范围应该如何理解?
初学Python,自己尝试着写了一个爬虫,主要代码如下import json import scrapy from bs4 import BeautifulSoup from docx import ...
- C++ 私有成员变量的理解
私有成员变量的概念,在脑海中的现象是,以private关键字声明,是类的实现部分,不对外公开,不能在对象外部访问对象的私有成员变量. 然而,在实现拷贝构造函数和赋值符函数时,在函数里利用对象直接访问了 ...
- 【数据收集】名义变量、序级变量、区间变量、比率变量的理解及例子
[数据收集]名义变量.序级变量.区间变量.比率变量的理解及例子 名义变量(Nominal Variable) 分类及举例 序级变量(Ordinal Variable) 分类和举例 区间变量(Inter ...
- 《深入理解mybatis原理》 MyBatis缓存机制的设计与实现
本文主要讲解MyBatis非常棒的缓存机制的设计原理,给读者们介绍一下MyBatis的缓存机制的轮廓,然后会分别针对缓存机制中的方方面面展开讨论. MyBatis将数据缓存设计成两级结构,分为一级缓存 ...
- [笔试题]交换两个数不使用第三方变量 深入理解按位异或运算符
异或运算相当与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 = ...
- 深入理解mybatis原理, Mybatis初始化SqlSessionFactory机制详解(转)
文章转自http://blog.csdn.net/l454822901/article/details/51829785 对于任何框架而言,在使用前都要进行一系列的初始化,MyBatis也不例外.本章 ...
最新文章
- Huggingface及BERT代码介绍
- 后盾网lavarel视频项目---vue实现动态添加和删除板块
- 如何成为一名大数据工程师?
- 多线程编程指南 part 2
- 谈判失败:Oracle杀死Java EE
- 特征工程系列学习(零)引言
- aes加密php源码,AES加解密类源码 · ThinkPHP5高阶实战教程 --诠释为API开发而生 · 看云...
- 《微服务设计》(一)---- 微服务
- Java常见面试题:对象的访问定位的两种方式
- C/C++(变量作用域)
- 乌卡时代下,企业供应链管理体系的应对策略
- 皮卡智能联手全球最大贸易服务商PingPong,共推AIGC应用落地服务
- css渐变描边视频,CSS制作渐变描边等文字特效
- 快递100API 手机H5使用
- crossorigin
- LK金字塔光流法与简单实现
- STM32操作24位AD芯片ADS1246
- 众筹,帮创业者跨过“死亡之谷”
- java使用jxls导出excel功能
- 彼得·德鲁克--管理思想摘录
热门文章
- Java一个简单的爬虫:爬去网页代码
- 外链引入css有哪些方式_CSS 文件的4种引入方式
- javaWeb三大框架总结
- python查看对象占用内存_『Python』内存分析_List对象内存占用分析
- java 多线程 选择题_Java多线程之三道多线程练习题
- git rollback代码都没了_ECBM库也能自动更新吗?——论GIT的用法
- ntp协议原理linux网络编程,NTP协议
- 服务器u盘安装win7系统,如何用U盘安装win7原版64位系统
- linux启用shell脚本,linux下开机启动shell脚本
- OpenShift 4.3 通过脚本配置ServiceMesh和Serverless运行环境