PHP 全局变量 - 超全局变量
PHP 中的许多预定义变量都是“超全局的”,这意味着它们在一个脚本的全部作用域中都可用。在函数或方法中无需执行 global $variable; 就可以访问它们。
这些超全局变量是:
- $GLOBALS
- $_SERVER
- $_REQUEST
- $_POST
- $_GET
- $_FILES
- $_ENV
- $_COOKIE
- $_SESSION
$GLOBALS — 引用全局作用域中可用的全部变量
$GLOBALS 这种全局变量用于在 PHP 脚本中的任意位置访问全局变量(从函数或方法中均可)。
PHP 在名为 $GLOBALS[index] 的数组中存储了所有全局变量。变量的名字就是数组的键。
<?php
$x = 75;
$y = 25;
function addition() {
$GLOBALS['z'] = $GLOBALS['x'] + $GLOBALS['y'];
}
addition();
echo $z; //100
?>
在上面的例子中,由于 z 是 $GLOBALS 数组中的变量,因此在函数之外也可以访问它
$_REQUEST
PHP $_REQUEST 用于收集 HTML 表单提交的数据。 $_REQUEST作用相当于$_POST+$_GET+$_COOKIE
$_ENV 为空的原因及解决办法
variables_order = "EGPCS"
上述php.ini配置表示 PHP 接受的外部变量来源及顺序,EGPCS 是 Environment、Get、Post、Cookies 和 Server 的缩写。如果 variables_order 的配置中缺少 E ,则 PHP 无法接受环境变量,那么 $_ENV 也就为空了。
$_SERVER
$_SERVER 这种超全局变量保存关于报头、路径和脚本位置的信息。
- $_SERVER['PHP_SELF'] #当前正在执行脚本的文件名,包含文件夹路径与 document root相关。
- $_SERVER['argv'] #传递给该脚本的参数,返回数组形式
- $_SERVER['argc'] #包含传递给程序的命令行参数的个数(如果运行在命令行模式)。
- $_SERVER['GATEWAY_INTERFACE'] #服务器使用的 CGI 规范的版本。例如,“CGI/1.1”。
- $_SERVER['SERVER_NAME'] #当前运行脚本所在服务器主机的名称。
- $_SERVER['SERVER_SOFTWARE'] #服务器标识的字串,在响应请求时的头部中给出。
- $_SERVER['SERVER_PROTOCOL'] #请求页面时通信协议的名称和版本。例如,“HTTP/1.0”。
- $_SERVER['REQUEST_METHOD'] #访问页面时的请求方法。例如:“GET”、“HEAD”,“POST”,“PUT”。
- $_SERVER['QUERY_STRING'] #查询(query)的字符串,也就是传递的参数和值的url.
- $_SERVER['DOCUMENT_ROOT'] #当前运行脚本所在的文档根目录。在服务器配置文件中定义,类似于网站的WWW根目录。
- $_SERVER['HTTP_ACCEPT'] #当前请求的 Accept: 头部的内容。
- $_SERVER['HTTP_ACCEPT_CHARSET'] #当前请求的 Accept-Charset: 头部的内容。例如:“iso-8859-1,*,utf-8”。
- $_SERVER['HTTP_ACCEPT_ENCODING'] #当前请求的 Accept-Encoding: 头部的内容。例如:“gzip”。
- $_SERVER['HTTP_ACCEPT_LANGUAGE']#当前请求的 Accept-Language: 头部的内容。例如:“en”。
- $_SERVER['HTTP_CONNECTION'] #当前请求的 Connection: 头部的内容。例如:“Keep-Alive”。
- $_SERVER['HTTP_HOST'] #当前请求的 Host: 头部的内容,和$_SERVER['SERVER_NAME']类似,但是这里实际上是有80端口的,也就是说:$_SERVER['HTTP_HOST'] = $_SERVER['SERVER_NAME']:$_SERVER['SERVER_PORT'];
- $_SERVER['HTTP_REFERER'] #链接到当前页面的前一页面的 URL 地址。
- $_SERVER['HTTP_USER_AGENT'] #当前请求的 User_Agent: 头部的内容。
- $_SERVER['HTTPS'] — 如果通过https访问,则被设为一个非空的值(on),否则返回off
- $_SERVER['REMOTE_ADDR'] #正在浏览当前页面用户的IP 地址。
- $_SERVER['SERVER_ADDR' ]:当前运行脚本所在的服务器的 IP 地址
- $_SERVER['PATH_INFO'] http://www.test.com/index.php/foo/bar.html?c=index&m=search
- $_SERVER['REMOTE_HOST'] #正在浏览当前页面用户的主机名。
- $_SERVER['REMOTE_PORT'] #用户连接到服务器时所使用的端口。
- $_SERVER['SCRIPT_FILENAME'] #当前执行脚本的绝对路径名。
- $_SERVER['SERVER_ADMIN'] #管理员信息
- $_SERVER['SERVER_PORT'] #服务器所使用的端口
- $_SERVER['SERVER_SIGNATURE'] #包含服务器版本和虚拟主机名的字符串。
- $_SERVER['PATH_TRANSLATED'] #当前脚本所在文件系统(不是文档根目录)的基本路径。
- $_SERVER['SCRIPT_NAME'] #包含当前脚本的路径。这在页面需要指向自己时非常有用。
- $_SERVER['REQUEST_URI'] #访问此页面所需的 URI。例如,“/index.html”。
- $_SERVER['PHP_AUTH_USER'] #当 PHP 运行在 Apache 模块方式下,并且正在使用 HTTP 认证功能,这个变量便是用户输入的用户名。
- $_SERVER['PHP_AUTH_PW'] #当 PHP 运行在 Apache 模块方式下,并且正在使用 HTTP 认证功能,这个变量便是用户输入的密码。
- $_SERVER['AUTH_TYPE'] #当 PHP 运行在 Apache 模块方式下,并且正在使用 HTTP 认证功能,这个变量便是认证的类型。
魔术常量(Magic constants)
PHP中的常量大部分都是不变的,但是有8个常量会随着他们所在代码位置的变化而变化,这8个常量被称为魔术常量。
__LINE__
,文件中的当前行号__FILE__
,文件的完整路径和文件名__DIR__
,文件所在的目录__FUNCTION__
,函数名称__CLASS__
,类的名称__TRAIT__
,Trait的名字__METHOD__
,类的方法名__NAMESPACE__
,当前命名空间的名称
这些魔术常量常常被用于获得当前环境信息或者记录日志。
魔术方法
PHP中把以两个下划线__
开头的方法称为魔术方法,这些方法在PHP中充当了举足轻重的作用。 魔术方法包括:
__construct()
, 实例化对象时被调用__destruct()
,当删除一个对象或对象操作终止时被调用__call()
,在对象中调用一个不存在的方法时调用__callStatic()
,用静态方式中调用一个不存在的方法时调用__get()
,获得一个类的成员变量时调用__set()
,设置一个类的成员变量时调用__isset()
,当对不存在的属性调用isset()
或empty()
时调用__unset()
,当对不存在的属性调用unset()
时被调用。__sleep()
,执行serialize()
时,先会调用这个函数__wakeup()
,执行unserialize()
时,先会调用这个函数__toString()
,打印类的时候调用 echo class__invoke()
,调用函数的方式调用一个对象时的回应方法__set_state()
,调用var_export()
导出类时,此静态方法会被调用。__clone()
,当对象复制完成时调用- __autoload() 实例化一个对象时,如果对应的类不存在,则该方法被调用
- __debugInfo() 调用
var_dump()
时,此静态方法会被调用。
__construct()
和__destruct()
构造函数和析构函数应该不陌生,他们在对象创建和消亡时被调用。例如我们需要打开一个文件,在对象创建时打开,对象消亡时关闭
<?php
class FileRead
{
protected $handle = NULL;
function __construct() {
$this->handle = fopen();
}
function __destruct() {
fclose($this->handle);
}
}
?>
__call()
和__callStatic()
在对象中调用一个不可访问方法时会调用这两个方法,后者为静态方法。这两个方法我们在可变方法(Variable functions)调用中可能会用到。
<?php
class MethodTest
{
public function __call($name, $arguments) {
echo "Calling object method '$name' " . implode(', ', $arguments) . "\n";
}
public static function __callStatic($name, $arguments) {
echo "Calling static method '$name' " . implode(', ', $arguments) . "\n";
}
}
$obj = new MethodTest;
$obj->runTest('in object context');
MethodTest::runTest('in static context');
?>
__get()
,__set()
,__isset()
和__unset()
当get/set一个类的成员变量时调用这两个函数。例如我们将对象变量保存在另外一个数组中,而不是对象本身的成员变量
<?php
class MethodTest
{
private $data = array();
public function __set($name, $value) {
$this->data[$name] = $value;
}
public function __get($name) {
$object_name = null;
if (array_key_exists($name, $this->data)) {
$object_name = $this->data[$name];
}
return $object_name;
}
public function __isset($name) {
return isset($this->data[$name]);
}
public function __unset($name) {
unset($this->data[$name]);
}
}
?>
__sleep()
和__wakeup()
当我们在执行
serialize()
和
unserialize()
时,会先调用这两个函数。例如我们在序列化一个对象时,这个对象有一个数据库链接,想要在反序列化中恢复链接状态,则可以通过重构这两个函数来实现链接的恢复。serialize() 函数会检查类中是否存在一个魔术方法 __sleep()。unserialize() 会检查是否存在一个 __wakeup() 方法
__toString()
对象当成字符串时的回应方法。例如使用echo $obj;
来输出一个对象
<?php
namespace DesignPatterns\Structural\FluentInterface;
class Sql
{
/**
* @var array
*/
private $fields = [];
/**
* @var array
*/
private $from = [];
/**
* @var array
*/
private $where = [];
public function select(array $fields): Sql
{
$this->fields = $fields;
return $this;
}
public function from(string $table, string $alias): Sql
{
$this->from[] = $table.' AS '.$alias;
return $this;
}
public function where(string $condition): Sql
{
$this->where[] = $condition;
return $this;
}
public function __toString(): string
{
return sprintf(
'SELECT %s FROM %s WHERE %s',
join(', ', $this->fields),
join(', ', $this->from),
join(' AND ', $this->where)
);
}
}
echo (new Sql()) ->select(['foo', 'bar'])->from('foobar', 'f')->where('f.bar = 1');
这个方法只能返回字符串,而且不可以在这个方法中抛出异常,否则会出现致命错误。
__invoke()
调用函数的方式调用一个对象时的回应方法。如下
<?php
class CallableClass
{
function __invoke($x) {
var_dump($x);
}
}
$obj = new CallableClass;
$obj(5);
var_dump(is_callable($obj));
?>
__set_state()
调用var_export()
导出类时,此静态方法会被调用。
__clone()
当对象复制完成时调用。例如在设计模式详解及PHP实现:单例模式一文中提到的单例模式实现方式,利用这个函数来防止对象被克隆。
<?php
class Test
{
public function __clone() {
echo "clone call me";
}
}
$t = new Test();
$t1 = clone $t;