PHP静态字段是真正的静态字段吗?
我觉得PHP静态字段仅在整个请求中是静态的 我的控制器类中有以下代码:PHP静态字段是真正的静态字段吗?,php,Php,我觉得PHP静态字段仅在整个请求中是静态的 我的控制器类中有以下代码: static $a = 2; public function debug() { var_dump(self::$a++); var_dump(self::$a++); } 无论我请求调试多少次,它都会输出: int 2 int 3 与我对java中的静态的了解非常不同。是的,PHP中的静态是“真正的”静态 您观察到的是PHP和Java中不同应用程序生命周期的结果 在Java中,web应用程序在WebS
static $a = 2;
public function debug()
{
var_dump(self::$a++);
var_dump(self::$a++);
}
无论我请求调试多少次,它都会输出:
int 2
int 3
与我对java中的静态的了解非常不同。是的,PHP中的静态是“真正的”静态
您观察到的是PHP和Java中不同应用程序生命周期的结果
在Java中,web应用程序在WebServer(HTTP服务器)内部运行,在初始类加载后,在后续请求中,该应用程序将重用已加载的内容。因此,类(和静态属性)初始化在应用程序生命周期中只发生一次
对于典型的PHP web应用程序,它看起来有点不同
HTTP服务器是独立的应用程序,它侦听HTTP请求并按需运行PHP(并非所有HTTP请求都必须传递给PHP)。
PHP作为一个单独的进程运行,请求被传递,在收到应答后,进程被丢弃。每个请求都由完全独立的流程处理。类(和静态属性)每次都从零开始加载和初始化
下面是用PHP编写的简单(非常)HTTP服务器,它将模拟Java Web服务器的工作方式
<?php
class Server {
private $socket;
private $routes = [];
public function __construct($address, $port, $backlog = 5) {
$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
if ($socket === false) {
throw new Exception("socket_create() failed: reason: " . socket_strerror(socket_last_error($socket)));
}
if (socket_bind($socket, $address, $port) === false) {
throw new Exception("socket_bind() failed: reason: " . socket_strerror(socket_last_error($socket)));
}
if (socket_listen($socket, $backlog) === false) {
throw new Exception("socket_listen() failed: reason: " . socket_strerror(socket_last_error($socket)));
}
$this->socket = $socket;
}
public function listen() {
while( ($requestSocket = socket_accept($this->socket)) !== false ) {
$this->handleRequestSocket($requestSocket);
}
throw new Exception("socket_accept() failed: reason: " . socket_strerror(socket_last_error($this->socket)));
}
public function registerController($url, $controller) {
$this->routes[$url] = $controller;
}
private function handleRequestSocket($socket) {
$buffer = "";
while(false !== ($part = socket_read($socket, 1024, PHP_NORMAL_READ))){
$buffer .= $part;
if(substr($buffer, -4) == "\r\n\r\n") break;
}
$buffer = trim($buffer);
echo "\n======\n$buffer\n======\n";
$response = $this->handleRequest($buffer);
if (null === $response){
socket_write($socket, "HTTP/1.1 404 Not Found\r\nConnection: close\r\n\r\n");
} else {
socket_write($socket, "HTTP/1.1 200 OK\r\nContent-Length: ".strlen($response)."\r\n\r\n$response");
}
socket_close($socket);
}
private function handleRequest($raw) {
$lines = explode("\r\n", $raw);
$req = explode(" ", $lines[0]);
$method = $req[0];
$url = $req[1];
if(isset($this->routes[$url])) {
return (string) (is_callable($this->routes[$url]) ? $this->routes[$url]($raw) : $this->routes[$url]);
}
return null;
}
}
class ControllerWithStatic {
private static $static = 0;
public function handle() {
return "Hello from static: " . (self::$static++) . "\n";
}
}
$server = new Server($argv[1], $argv[2]);
$c = new ControllerWithStatic();
$server->registerController("/", "Hello world\n");
$server->registerController("/closure", function(){return "Hello world from closure\n";});
$server->registerController("/static", [$c, 'handle']);
$server->registerController("/static2", function(){
return (new ControllerWithStatic())->handle();
});
$server->listen();
e、 g
现在在浏览器中打开http://127.0.0.1:8080/static
或http://127.0.0.1:8080/static2
您将获得
Hello from static: 0
Hello from static: 1
Hello from static: 2
...
只要不重新启动服务器,这个数字就会增加。静态并不意味着恒定。这意味着它属于该类,而不是该类的实例化对象。至于“在整个请求中仅仅是静态的”:每个请求都是一个新的PHP实例/线程,如果您想在请求之间保留变量,您必须使用会话($\u session
)或某种其他存储(数据库、文件等)。PHP中的静态属性与Java中的完全相同。@amruths。我认为这些是不同的问题。ccKep提到的“每个请求都是PHP的一个新实例/线程”,可能就是我的问题的答案。@bijiDang。这是正确的,根据WEB服务器,每一个新的请求都将成为服务器端语言(如PHP)的新线程。。。这就是为什么输出像你提到的那样。。。您可以使用会话共享变量的值,也可以将变量保存到数据库中并检索回新请求。
php server.php 127.0.0.1 8080
Hello from static: 0
Hello from static: 1
Hello from static: 2
...