PHP8 FAQ:命名空間必知必會

2023-08-18 13:38 更新

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

本文分兩節(jié):常見問題、有助于完全理解的實現(xiàn)詳情。

首先,常見問題。

  1. 如果我不用命名空間,是否需要關心它?
  2. 我如何在命名空間內使用一個全局/內置的類?
  3. 如何在命名空間內訪問它自己的類、函數(shù)、常量?
  4. 像 \my\name 和 \name 這樣的名稱是如何解析的?
  5. 像 my\name 這樣的名稱是如何解析的?
  6. 像 name 這樣的非限定類名是如何解析的?
  7. 像 name 這樣的非限定常量和函數(shù)名是如何解析的?

為了幫助理解,我們提供了一些命名空間實現(xiàn)細節(jié)。

  1. 在同一個文件中,導入名稱不能和定義的類名發(fā)生沖突。
  2. 不允許嵌套 namespace。
  3. 動態(tài)命名空間名稱(引號標識)應該轉義反斜線。
  4. 引用一個未定義的、帶反斜線的常量,會導致 fatal 錯誤并退出
  5. 不能重載特殊常量: NULL、TRUE、FALSE、ZEND_THREAD_SAFE、ZEND_DEBUG_BUILD

如果我不用命名空間,是否需要關心它?

不需要。命名空間不影響現(xiàn)存的代碼,也不影響即將要寫下的不含命名空間的代碼。 想要的話可以這樣寫:

示例 #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
class MyException extends \Exception {}
?>

如何在命名空間內訪問它自己的類、函數(shù)、常量?

示例 #4 在命名空間中訪問內置的類、函數(shù)、常量

<?php
namespace foo;

class MyClass {}

// 以當前命名空間中的 class 作為參數(shù)的類型
function test(MyClass $parameter_type_example = null) {}
// 以當前命名空間中的 class 作為參數(shù)的類型的另一種方式
function test(\foo\MyClass $parameter_type_example = null) {}

// 在當前命名空間中擴展一個類
class Extended extends MyClass {}

// 訪問全局函數(shù)
$a = \globalfunc();

// 訪問全局常量
$b = \INI_ALL;
?>

像 \my\name 和 \name 這樣的名稱是如何解析的?

以 \ 開頭的名稱總是會解析成原樣, 因此 \my\name 實際上是 my\name, 而 \Exception 是 Exception。

示例 #5 完全限定名稱

<?php
namespace foo;
$a = new \my\name(); // class "my\name" 的實例
echo \strlen('hi'); // 調用函數(shù) "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(); // class "foo\my\name" 的實例
foo\bar::name(); // 調用 class "blah\blah\bar" 的靜態(tài)方法 "name"
my\bar(); // 調用函數(shù) "foo\my\bar"
$a = my\BAR; // 設置 $a 的值為 "foo\my\BAR"
?>

像 name 這樣的非限定名稱是如何解析的?

像 name 這樣不包含反斜線的名稱, 能夠以兩種不同的方式解析。

如果有導入語句,設置別名為 name,就會應用導入別名。

如果沒有,就會把當前命名空間添加到 name 的前綴。

示例 #7 非限定類名

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

$a = new name(); // class "foo\name" 的實例
foo::name(); // 調用 class "blah\blah" 的靜態(tài)方法 "name"
?>

像 name 這樣的非限定常量和函數(shù)名是如何解析的?

像 name 這樣不包含反斜線的常量和函數(shù)名,能以兩種不同的方式解析。

首先,當前命名空間會添加到 name 的前綴。

然后,如果當前命名空間不存在函數(shù)和常量 name, 而全局存在,就會使用全局的函數(shù)和常量 name。

示例 #8 非限定函數(shù)和常量名

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

const FOO = 1;

function my() {}
function foo() {}
function sort(&$a)
{
sort($a);
$a = array_flip($a);
return $a;
}

my(); // 調用 "foo\my"
$a = strlen('hi'); // 由于 "foo\strlen" 不存在,所以調用全局的 "strlen"
$arr = array(1,3,2);
$b = sort($arr); // 調用函數(shù) "foo\sort"
$c = foo(); // 未導入,調用函數(shù) "foo\foo"

$a = FOO; // 未導入,設置 $a 為常量 "foo\FOO" 的值
$b = INI_ALL; // 設置 $b 為全局常量 "INI_ALL" 的值
?>

在同一個文件中,導入名稱不能和定義的類名發(fā)生沖突

允許以下腳本中的組合:

file1.php

<?php
namespace my\stuff;
class MyClass {}
?>

another.php

<?php
namespace another;
class thing {}
?>

file2.php

<?php
namespace my\stuff;
include 'file1.php';
include 'another.php';

use another\thing as MyClass;
$a = new MyClass; // class "thing" 的實例來自于命名空間 another
?>

盡管在 my\stuff 命名空間中存在 MyClass, 因為類定義在了獨立的文件中,所以不會發(fā)生名稱沖突。 不過,接下來的例子中,因為 MyClass 定義在了 use 語句的同一個文件中, 所以發(fā)生了名稱沖突,導致了 fatal 錯誤。

<?php
namespace my\stuff;
use another\thing as MyClass;
class MyClass {} // fatal error: MyClass conflicts with import statement
$a = new MyClass;
?>

不允許嵌套 namespace

PHP 不允許嵌套 namespace

<?php
namespace my\stuff {
namespace nested {
class foo {}
}
}
?>

實際上,它看上去像是這樣:

<?php
namespace my\stuff\nested {
class foo {}
}
?>

動態(tài)命名空間名稱(引號標識)應該轉義反斜線

重要的是,字符串中反斜線是一個轉義字符,因此在字符串中使用時,必須要寫兩遍。 否則就會在無意中造成一些后果:

示例 #9 在雙引號字符串中使用命名空間的危險性

<?php
$a = new "dangerous\name"; // 在雙引號字符串中,\n 是換行符!
$obj = new $a;

$a = new 'not\at\all\dangerous'; // 這里沒有問題
$obj = new $a;
?>

在單引號字符串中,使用反斜線是安全的。 但在最佳實踐中,我們仍然推薦為所有字符串統(tǒng)一轉義反斜線。

引用一個未定義的、帶反斜線的常量,會導致 fatal 錯誤并退出

像 FOO 這樣的非限定名稱常量,如果使用的時候還沒定義, 會產生一個 notice。PHP 會假設該常量的值是 FOO。 如果沒有找到包含反斜線的常量,無論是完全或者不完全限定的名稱,都會產生 fatal 錯誤。

示例 #10 未定義的常量

<?php
namespace bar;
$a = FOO; // 產生 notice - undefined constants "FOO" assumed "FOO";
$a = \FOO; // fatal error, undefined namespace constant FOO
$a = Bar\FOO; // fatal error, undefined namespace constant bar\Bar\FOO
$a = \Bar\FOO; // fatal error, undefined namespace constant Bar\FOO
?>

不能重載特殊常量:NULL、TRUE、FALSE、ZEND_THREAD_SAFE、ZEND_DEBUG_BUILD

在命名空間內定義特殊的內置常量,會導致 fatal 錯誤

示例 #11 未定義的常量

<?php
namespace bar;
const NULL = 0; // fatal error;
const true = 'stupid'; // 也是 fatal error;
// etc.
?>


以上內容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號