[Related learning recommendations: php programming (video)]
Managing global state
Some global space is always needed in imperative languages. When programming PHP or extensions, we will make a clear distinction between what we call request-bound global variables and true global variables.
Request global variables are global variables that need to carry and remember information during request processing. A simple example is when you ask the user to provide a value in a function parameter and want to be able to use it in other functions. Except that this information "holds its value" across several PHP function calls, it only retains the value for the current request. The next request that comes should know nothing. PHP provides a mechanism to manage request global variables regardless of the multiprocessing model chosen, which we will cover in detail later in this chapter.
Real global variables are pieces of information that are retained across requests. This information is usually read-only. If you need to write to such a global variable as part of request handling, PHP can't help you. If you use threads as the multiprocessing model, you need to implement memory locking yourself. If you use processes as the multiprocessing model, you need to use your own IPC (inter-process communication). However, this should not happen in PHP extension programming.
Manage request global variables
Here is a simple extension example using request globals:
/* 真正的 C 全局 */ static zend_long rnd = 0; static void pib_rnd_init(void) { /* 在 0 到 100 之间随机一个数字 */ php_random_int(0, 100, &rnd, 0); } PHP_RINIT_FUNCTION(pib) { pib_rnd_init(); return SUCCESS; } PHP_FUNCTION(pib_guess) { zend_long r; if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &r) == FAILURE) { return; } if (r == rnd) { /* 将数字重置以进行猜测 */ pib_rnd_init(); RETURN_TRUE; } if (r <p>As you can see, this extension picks a random integer at the beginning of the request number, and then you can try to guess this array through <code>pib_guess()</code>. Once guessed, the number will reset. If the user wants to manually reset the number, it can also manually call <code>pib_reset()</code> to reset the value. <br>The random number is implemented as <em>a C global variable</em>. If PHP is used in-process as part of a multi-process model it is no longer an issue, if threads are used later, <strong>this is not okay</strong>. </p><blockquote> <p>NOTE</p> <p>As a reminder, you do not need to know which multi-process model you are going to use. When you design an extension, you must be prepared for both models. </p> </blockquote><p>When using threads, a C global variable is shared for each thread in the server. For example, in our example above, each concurrent user of the network server will share the same value. Some may reset the value initially, while others try to guess it. In short, you clearly understand the key issue with threads. </p><p>We must persist data to the same request. Even if running PHP's multi-process model will utilize threads, it must be <strong>bound</strong> to the current request. </p><h3 id="Use-TSRM-macros-to-protect-global-space">Use TSRM macros to protect global space</h3><p>PHP is designed with a layer that helps extension and kernel developers handle global requests. This layer is called <strong>TSRM</strong> (Thread-Safe Resource Management) and is exposed as a set of macros that you must use any time you need to access the request-bound globals (read and write). </p><p>In the case of a multi-process model using processes, behind the scenes, these macros will be parsed into code similar to the one we showed above. As we can see, the above code is completely valid if threading is not used. So, when using processes, these macros will be expanded to similar macros. </p><p>The first thing you have to do is declare a structure that will be the root of all your global variables: </p><pre class="brush:php;toolbar:false">ZEND_BEGIN_MODULE_GLOBALS(pib) zend_long rnd; ZEND_END_MODULE_GLOBALS(pib) /* 解析为 : * * typedef struct _zend_pib_globals { * zend_long rnd; * } zend_pib_globals; */
Then, create a global variable like this:
ZEND_DECLARE_MODULE_GLOBALS(pib) /* 解析为 zend_pib_globals pib_globals; */
Now, you can access the data using global macro accessors. This macro is created by the framework and should be defined in your php_pib.h header file. This looks like this:
#ifdef ZTS #define PIB_G(v) ZEND_MODULE_GLOBALS_ACCESSOR(pib, v) #else #define PIB_G(v) (pib_globals.v) #endif
As you can see, if ZTS mode is not enabled, i.e. compiling non-thread-safe PHP and extensions (we call it NTS mode: non-threaded safe), the macro simply resolves to the data declared in the structure. Therefore, there are the following changes:
static void pib_rnd_init(void) { php_random_int(0, 100, &PIB_G(rnd), 0); } PHP_FUNCTION(pib_guess) { zend_long r; if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &r) == FAILURE) { return; } if (r == PIB_G(rnd)) { pib_rnd_init(); RETURN_TRUE; } if (r <blockquote> <p>Note</p> <p>When using a process model, <em>TSRM</em> macros resolve to accesses to C global variables. </p> </blockquote><p>When using threads, i.e. when you compile ZTS PHP, things get more complicated. Then, all the macros we see resolve to something completely different, which is hard to explain here. Basically, <em>TSRM</em> does a hard job using TLS (Thread Local Storage) when compiled with ZTS. </p><blockquote> <p>NOTE</p> <p>In short, when compiling in ZTS, global variables are bound to the current thread. When compiling NTS, global variables will be bound to the current process. The TSRM macro handles the hard work. You may be interested in how this works, and browse the /TSRM directory of the PHP source code to learn more about PHP thread safety. </p> </blockquote><h3 id="Using-global-hooks-in-extensions">Using global hooks in extensions</h3><p>Sometimes, it may be necessary to initialize a global variable to some default value, usually zero. The TSRM system with the help of the engine provides a hook to provide default values for your global variables, we call it <strong>GINIT</strong>. </p><blockquote> <p>Notice</p> <p>关于 PHP 挂钩的完整信息,请参考 PHP 生命周期章节。</p> </blockquote><p>让我们将随机值设为零:</p><pre class="brush:php;toolbar:false">PHP_GSHUTDOWN_FUNCTION(pib) { } PHP_GINIT_FUNCTION(pib) { pib_globals->rnd = 0; } zend_module_entry pib_module_entry = { STANDARD_MODULE_HEADER, "pib", NULL, NULL, NULL, NULL, NULL, NULL, "0.1", PHP_MODULE_GLOBALS(pib), PHP_GINIT(pib), PHP_GSHUTDOWN(pib), NULL, /* PRSHUTDOWN() */ STANDARD_MODULE_PROPERTIES_EX };
我们选择仅显示 zend_module_entry
(和其他 NULL
)的相关部分。如你所见,全局管理挂钩发生在结构的中间。首先是PHP_MODULE_GLOBALS()
来确定全局变量的大小,然后是我们的 GINIT
和 GSHUTDOWN
钩子。然后我们使用了STANDARD_MODULE_PROPERTIES_EX
关闭结构,而不是STANDARD_MODULE_PROPERTIES
。只需以正确的方式完成结构即可,请参阅?:
#define STANDARD_MODULE_PROPERTIES NO_MODULE_GLOBALS, NULL, STANDARD_MODULE_PROPERTIES_EX
在GINIT
函数中,你传递了一个指向全局变量当前存储位置的指针。你可以使用它来初始化全局变量。在这里,我们将零放入随机值(虽然不是很有用,但我们接受它)。
警告
不要在 GINIT 中使用
PIB_G()
宏。使用你得到的指针。注意
对于当前进程,在
MINIT()
之前启动了GINIT()
。如果是 NTS,就这样而已。 如果是 ZTS,线程库产生的每个新线程都会额外调用GINIT()
。警告
GINIT()
不作为RINIT()
的一部分被调用。如果你需要在每次新请求时清除全局变量,则需要像在本章所示示例中所做的那样手动进行。
完整的例子
这是一个更高级的完整示例。如果玩家获胜,则将其得分(尝试次数)添加到可以从用户区获取的得分数组中。没什么难的,得分数组在请求启动时初始化,然后在玩家获胜时使用,并在当前请求结束时清除:
ZEND_BEGIN_MODULE_GLOBALS(pib) zend_long rnd; zend_ulong cur_score; zval scores; ZEND_END_MODULE_GLOBALS(pib) ZEND_DECLARE_MODULE_GLOBALS(pib) static void pib_rnd_init(void) { /* 重置当前分数 */ PIB_G(cur_score) = 0; php_random_int(0, 100, &PIB_G(rnd), 0); } PHP_GINIT_FUNCTION(pib) { /* ZEND_SECURE_ZERO 是 memset(0)。也可以解析为 bzero() */ ZEND_SECURE_ZERO(pib_globals, sizeof(*pib_globals)); } ZEND_BEGIN_ARG_INFO_EX(arginfo_guess, 0, 0, 1) ZEND_ARG_INFO(0, num) ZEND_END_ARG_INFO() PHP_RINIT_FUNCTION(pib) { array_init(&PIB_G(scores)); pib_rnd_init(); return SUCCESS; } PHP_RSHUTDOWN_FUNCTION(pib) { zval_dtor(&PIB_G(scores)); return SUCCESS; } PHP_FUNCTION(pib_guess) { zend_long r; if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &r) == FAILURE) { return; } if (r == PIB_G(rnd)) { add_next_index_long(&PIB_G(scores), PIB_G(cur_score)); pib_rnd_init(); RETURN_TRUE; } PIB_G(cur_score)++; if (r <p>这里必须要注意的是,如果你希望在请求之间保持分数,PHP 不提供任何便利。而是需要一个持久的共享存储,例如文件,数据库,某些内存区域等。PHP 的设计目的不是将信息持久存储在其内部的请求,因此它不提供这么做,但它提供了实用程序来访问请求绑定的全局空间,如我们所示。</p><p>然后,很容易地在<code>RINIT()</code>中初始化一个数组,然后在<code>RSHUTDOWN()</code>中销毁它。请记住,<code>array_init</code>创建一个zend_array 并放入一个 zval。但这是免分配的,不要担心分配用户无法使用的数组(因此浪费分配),<code>array_init()</code>非常廉价 (阅读源代码)。</p><p>当我们将这样的数组返回给用户时,我们不会忘记增加其引用计数(在 <code>RETVAL_ZVAL</code>中),因为我们在扩展中保留了对此类数组的引用。</p><h2 id="使用真实的全局变量">使用真实的全局变量</h2><p>真实全局变量是非线程保护的真实C全局变量。有时可能会需要它们。但是请记住主要规则:在处理请求时,不能安全地写入此类全局变量。因此,通常在 PHP 中,我们需要此类变量并将其用作只读变量。</p><p>请记住,在 PHP 生命周期的<code>MINIT()</code>或<code>MSHUTDOWN()</code>步骤中编写真实全局变量是绝对安全的。但是不能在处理请求时给他们写入值(但可以从他们那里读取)。</p><p>因此,一个简单的示例是你想要读取环境值以对其进行处理。此外,初始化持久性的 zend_string并在之后处理某些请求时加以利用是很常见的。</p><p>这是介绍真实全局变量的修补示例,我们仅显示与先前代码的差异,而不显示完整代码:</p><pre class="brush:php;toolbar:false">static zend_string *more, *less; static zend_ulong max = 100; static void register_persistent_string(char *str, zend_string **result) { *result = zend_string_init(str, strlen(str), 1); zend_string_hash_val(*result); GC_FLAGS(*result) |= IS_INTERNED; } static void pib_rnd_init(void) { /* 重置当前分数 */ PIB_G(cur_score) = 0; php_random_int(0, max, &PIB_G(rnd), 0); } PHP_MINIT_FUNCTION(pib) { char *pib_max; register_persistent_string("more", &more); register_persistent_string("less", &less); if (pib_max = getenv("PIB_RAND_MAX")) { if (!strchr(pib_max, '-')) { max = ZEND_STRTOUL(pib_max, NULL, 10); } } return SUCCESS; } PHP_MSHUTDOWN_FUNCTION(pib) { zend_string_release(more); zend_string_release(less); return SUCCESS; } PHP_FUNCTION(pib_guess) { zend_long r; if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &r) == FAILURE) { return; } if (r == PIB_G(rnd)) { add_next_index_long(&PIB_G(scores), PIB_G(cur_score)); pib_rnd_init(); RETURN_TRUE; } PIB_G(cur_score)++; if (r <p>在这里我们创建了两个 zend_string 变量 <code>more</code> 和 <code>less</code>。这些字符串不需要像以前一样在使用时立即创建和销毁。这些是不可变的字符串,只要保持不变,就可以分配一次并在需要的任何时间重复使用(即只读)。我们在<code>zend_string_init()</code>中使用持久分配,在<code>MINIT()</code>中初始化这两个字符串,我们现在预先计算其哈希值(而不是先执行第一个请求),并且我们告诉 zval 垃圾收集器,这些字符串已被扣留,因此它将永远不会尝试销毁它们(但是,如果将它们用作写操作(例如连接)的一部分,则可能需要复制它们)。显然我们不会忘记在<code>MSHUTDOWN()</code>中销毁这些字符串。</p><p>然后在<code>MINIT()</code>中我们探查一个<code>PIB_RAND_MAX</code>环境,并将其用作随机数选择的最大范围值。由于我们使用无符号整数,并且我们知道<code>strtoull()</code>不会抱怨负数(因此将整数范围包裹为符号不匹配),我们只是避免使用负数(经典的libc解决方法)。</p><p style="position: absolute!important;height: 1px;width: 1px;overflow: hidden;clip: rect(1px,1px,1px,1px);"><br></p><blockquote><p> If you want to know more about programming learning, please pay attention to the <a href="https://www.php.cn/" target="_blank">php training</a> column! <br></p></blockquote>
The above is the detailed content of PHP global management method. For more information, please follow other related articles on the PHP Chinese website!

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\ \;||\xc2\xa0)/","其他字符",$str)”语句。

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

VSCode Windows 64-bit Download
A free and powerful IDE editor launched by Microsoft

PhpStorm Mac version
The latest (2018.2.1) professional PHP integrated development tool

DVWA
Damn Vulnerable Web App (DVWA) is a PHP/MySQL web application that is very vulnerable. Its main goals are to be an aid for security professionals to test their skills and tools in a legal environment, to help web developers better understand the process of securing web applications, and to help teachers/students teach/learn in a classroom environment Web application security. The goal of DVWA is to practice some of the most common web vulnerabilities through a simple and straightforward interface, with varying degrees of difficulty. Please note that this software

Zend Studio 13.0.1
Powerful PHP integrated development environment

EditPlus Chinese cracked version
Small size, syntax highlighting, does not support code prompt function
