2024 年 PHP 日本大会

常见问题解答:你需要了解的关于命名空间的事情

(PHP 5 >= 5.3.0, PHP 7, PHP 8)

本常见问题解答分为两部分:常见问题,以及一些有助于充分理解的实现细节。

首先,是常见问题。

  1. 如果我不使用命名空间,我需要关心这些吗?
  2. 如何在命名空间中使用内部或全局类?
  3. 如何在它们自己的命名空间中使用命名空间类、函数或常量?
  4. \my\name\name 这样的名称是如何解析的?
  5. my\name 这样的名称是如何解析的?
  6. name 这样的非限定类名是如何解析的?
  7. name 这样的非限定函数名或非限定常量名是如何解析的?

有一些命名空间实现的实现细节有助于理解。

  1. 导入名称不能与在同一文件中定义的类冲突。
  2. 不允许嵌套命名空间。
  3. 动态命名空间名称(带引号的标识符)应转义反斜杠。
  4. 使用任何反斜杠引用的未定义常量将导致致命错误
  5. 无法覆盖特殊常量 nulltruefalse

如果我不使用命名空间,我需要关心这些吗?

不会。命名空间不会以任何方式影响任何现有代码,也不会影响任何尚未编写的、不包含命名空间的代码。您可以根据需要编写此代码

示例 #1 在命名空间之外访问全局类

<?php
$a
= new \stdClass;
?>

这在功能上等效于

示例 #2 在命名空间之外访问全局类

<?php
$a
= new stdClass;
?>

如何在命名空间中使用内部或全局类?

示例 #3 在命名空间中访问内部类

<?php
namespace foo;
$a = new \stdClass;

function
test(\ArrayObject $parameter_type_example = null) {}

$a = \DirectoryIterator::CURRENT_AS_FILEINFO;

// 扩展内部或全局类
class MyException extends \Exception {}
?>

如何在它们自己的命名空间中使用命名空间类、函数或常量?

示例 #4 在命名空间中访问内部类、函数或常量

<?php
namespace foo;

class
MyClass {}

// 使用来自当前命名空间的类作为参数类型
function test(MyClass $parameter_type_example = null) {}
// 另一种使用来自当前命名空间的类作为参数类型的方法
function test(\foo\MyClass $parameter_type_example = null) {}

// 扩展来自当前命名空间的类
class Extended extends MyClass {}

// 访问全局函数
$a = \globalfunc();

// 访问全局常量
$b = \INI_ALL;
?>

\my\name\name 这样的名称是如何解析的?

\ 开头的名称始终解析为它们看起来的样子,因此 \my\name 事实上是 my\name,而 \ExceptionException

示例 #5 完全限定名称

<?php
namespace foo;
$a = new \my\name(); // 实例化 "my\name" 类
echo \strlen('hi'); // 调用函数 "strlen"
$a = \INI_ALL; // $a 设置为常量 "INI_ALL" 的值
?>

my\name 这样的名称是如何解析的?

包含反斜杠但不以反斜杠开头的名称,例如 my\name,可以通过两种不同的方式解析。

如果存在将另一个名称与 my 关联的导入语句,则将导入别名应用于 my\name 中的 my

否则,当前命名空间名称将添加到 my\name 的前面。

示例 #6 限定名称

<?php
namespace foo;
use
blah\blah as foo;

$a = new my\name(); // 实例化 "foo\my\name" 类
foo\bar::name(); // 调用类 "blah\blah\bar" 中的静态方法 "name"
my\bar(); // 调用函数 "foo\my\bar"
$a = my\BAR; // 将 $a 设置为常量 "foo\my\BAR" 的值
?>

name 这样的非限定类名是如何解析的?

不包含反斜杠的类名,例如 name,可以通过两种不同的方式解析。

如果存在将另一个名称与 name 关联的导入语句,则将应用导入别名。

否则,当前命名空间名称将添加到 name 的前面。

示例 #7 非限定类名

<?php
命名空间 foo;
use
blah\blah as foo;

$a = new name(); // 实例化 "foo\name" 类
foo::name(); // 调用 "blah\blah" 类中的静态方法 "name"
?>

name 这样的非限定函数名或非限定常量名是如何解析的?

不包含反斜杠的函数名或常量名,例如 name,可以通过两种不同的方式解析。

首先,将当前命名空间名称添加到 name 前面。

最后,如果当前命名空间中不存在常量或函数 name,则如果存在全局常量或函数 name,则使用它。

示例 #8 未限定的函数名或常量名

<?php
命名空间 foo;
use
blah\blah as foo;

const
FOO = 1;

function
my() {}
function
foo() {}
function
sort(&$a)
{
\sort($a); // 调用全局函数 "sort"
$a = array_flip($a);
return
$a;
}

my(); // 调用 "foo\my"
$a = strlen('hi'); // 调用全局函数 "strlen",因为 "foo\strlen" 不存在
$arr = array(1,3,2);
$b = sort($arr); // 调用函数 "foo\sort"
$c = foo(); // 调用函数 "foo\foo" - 导入不适用

$a = FOO; // 将 $a 设置为常量 "foo\FOO" 的值 - 导入不适用
$b = INI_ALL; // 将 $b 设置为全局常量 "INI_ALL" 的值
?>

导入名称不能与在同一文件中定义的类冲突。

以下脚本组合是合法的

file1.php

<?php
命名空间 my\stuff;
MyClass {}
?>

another.php

<?php
命名空间 another;
thing {}
?>

file2.php

<?php
命名空间 my\stuff;
include
'file1.php';
include
'another.php';

use
another\thing as MyClass;
$a = new MyClass; // 实例化来自 another 命名空间的 "thing" 类
?>

即使类 MyClass 存在于 my\stuff 命名空间中,也不会发生命名冲突,因为 MyClass 定义在单独的文件中。但是,下一个示例会导致致命错误,因为 MyClass 与 use 语句在同一个文件中定义。

<?php
命名空间 my\stuff;
use
another\thing as MyClass;
MyClass {} // 致命错误:MyClass 与导入语句冲突
$a = new MyClass;
?>

不允许嵌套命名空间。

PHP 不允许嵌套命名空间

<?php
命名空间 my\stuff {
命名空间
nested {
foo {}
}
}
?>
但是,很容易模拟嵌套命名空间,如下所示
<?php
命名空间 my\stuff\nested {
foo {}
}
?>

动态命名空间名称(带引号的标识符)应转义反斜杠

非常重要的是要意识到,由于反斜杠在字符串中用作转义字符,因此在字符串内使用时应始终将其加倍。否则,可能会产生意外的后果。

示例 #9 使用命名空间名称的双引号字符串中的危险

<?php
$a
= "dangerous\name"; // \n 在双引号字符串中是换行符!
$obj = new $a;

$a = 'not\at\all\dangerous'; // 这里没问题。
$obj = new $a;
?>
在单引号字符串中,反斜杠转义序列更安全,但仍然建议将所有字符串中的反斜杠转义作为最佳实践。

使用任何反斜杠引用的未定义常量将导致致命错误

任何未定义且未限定的常量(如 FOO)都会生成一条通知,解释 PHP 假设 FOO 是该常量的值。任何包含反斜杠的常量(限定的或完全限定的)如果找不到,都会产生致命错误。

示例 #10 未定义的常量

<?php
命名空间 bar;
$a = FOO; // 生成通知 - 未定义的常量 "FOO" 假设为 "FOO";
$a = \FOO; // 致命错误,未定义的命名空间常量 FOO
$a = Bar\FOO; // 致命错误,未定义的命名空间常量 bar\Bar\FOO
$a = \Bar\FOO; // 致命错误,未定义的命名空间常量 Bar\FOO
?>

不能覆盖特殊常量 nulltruefalse

任何尝试定义作为特殊内置常量的命名空间常量都会导致致命错误。

示例 #11 未定义的常量

<?php
命名空间 bar;
const
NULL = 0; // 致命错误;
const true = 'stupid'; // 也是致命错误;
//等等
?>

添加注释

用户贡献的注释 6 条注释

manolachef at gmail dot com
12 年前
有一种方法可以使用 define 函数并将第三个参数 case_insensitive 设置为 false 来定义作为特殊内置常量的命名空间常量。

<?php
命名空间 foo;
define(__NAMESPACE__ . '\NULL', 10); // 在当前命名空间中定义常量 NULL
var_dump(NULL); // 将显示 10
var_dump(null); // 将显示 NULL
?>

无需像通常那样在对 define() 的调用中指定命名空间
<?php
命名空间 foo;
define(INI_ALL, 'bar'); // 会产生警告 - 常量 INI_ALL 已定义。但是:

define(__NAMESPACE__ . '\INI_ALL', 'bar'); // 在当前命名空间中定义常量 INI_ALL
var_dump(INI_ALL); // 将显示 string(3)"bar"。到目前为止没有任何意外。但是:

define('NULL', 10); // 在当前命名空间中定义常量 NULL…
var_dump(NULL); // 将显示 10
var_dump(null); // 将显示 NULL
?>

如果参数 case_insensitive 设置为 true
<?php
命名空间 foo;
define (__NAMESPACE__ . '\NULL', 10, true); // 会产生警告 - 常量 null 已定义
?>
shaun at slickdesign dot com dot au
8年前
在使用变量从命名空间内部创建类或调用静态方法时,需要记住它们需要完整的命名空间才能使用相应的类;即使是在同一个命名空间内调用,也**不能**使用别名或简短名称。忽视这一点可能会导致代码使用错误的类,抛出致命错误的缺失类异常,或抛出错误或警告。

在这些情况下,可以使用魔术常量 __NAMESPACE__,或直接指定完整的命名空间和类名。函数 class_exists 也需要完整的命名空间和类名,并且可以用来确保不会由于缺少类而抛出致命错误。

<?php

命名空间 Foo;
Bar {
public static function
test() {
return
get_called_class();
}
}

命名空间
Foo\Foo;
Bar extends \Foo\Bar {
}

var_dump( Bar::test() ); // string(11) "Foo\Foo\Bar"

$bar = 'Foo\Bar';
var_dump( $bar::test() ); // string(7) "Foo\Bar"

$bar = __NAMESPACE__ . '\Bar';
var_dump( $bar::test() ); // string(11) "Foo\Foo\Bar"

$bar = 'Bar';
var_dump( $bar::test() ); // 致命错误:找不到类 'Bar' 或使用了不正确的类 \Bar
theking2 at king dot ma
2年前
就像类名一样,命名空间目前不区分大小写。因此这里不会显示错误

<?php declare(strict_types=1);
命名空间
Foo;
Bar {
public function
__construct() {
echo
'Map constructed';
}
}

$foobar = new \foo\bar();
teohad at NOSPAM dot gmail dot com
8年前
[编辑注:此行为是由 PHP 7.0 中的一个错误引起的,该错误已在 PHP 7.0.7 中修复。]

关于条目“导入名称不能与在同一文件中定义的类冲突”。
- 我发现从 PHP 7.0 开始不再是这样了。
在 PHP 7.0 中,您可以拥有一个名称与导入的类(或命名空间或同时两者)匹配的类。

<?php
命名空间 ns1 {
ns1 {
public static function
write() {
echo
"ns1\\ns1::write()\n";
}
}
}

命名空间
ns1\ns1 {
ns1c {
public static function
write() {
echo
"ns1\\ns1\\ns1c::write()\n";
}
}
}

命名空间
ns2 {
use
ns1\ns1 as ns1; // ns1 中的类和命名空间 ns1\ns1

// 下一个类在 php 5.6 中导致致命错误,在 7.0 中不会
ns1 {
public static function
write() {
echo
"ns2\\ns1::write()\n";
}
}

ns1::write(); // 调用导入的 ns1\ns1::write()
ns1\ns1c::write(); // 调用导入的 ns1\ns1\ns1c::write()
namespace\ns1::write(); // 调用 ns2\ns1::write()
}
?>
phpcoder
9年前
关于“函数和常量都不能通过 use 语句导入”。实际上,您可以在 PHP 5.6+ 中这样做。

<?php

// 导入函数 (PHP 5.6+)
use function My\Full\functionName;

// 为函数设置别名 (PHP 5.6+)
use function My\Full\functionName as func;

// 导入常量 (PHP 5.6+)
use const My\Full\CONSTANT;
?>
okaresz
11年前
更正 manolachef 的回答:define() 始终在全局命名空间中定义常量。

正如 nl-x at bita dot nl 在 https://php.net/manual/en/function.define.php, 的注释中所述,常量“NULL”可以用 define() 区分大小写地定义,但只能用 constant() 获取,使大写关键字 NULL 的含义成为 null 类型唯一的取值。
To Top