理解命名空间
在windows系统下同一个文件是不能存在一样同名的文件的,而在PHP开发中,如果在运行时,引入两个一样的类名,系统就会报错,而引入相同类名的情况是普遍存在的(以下将会有解释)。为了解决在同一脚本中引入一个相同类名会产生冲突的现象,PHP引入了命名空间的概念。PHP命名空间解决了在脚本中引入相同的类名产生冲突的问题。一种抽象的类比:我们这样理解:当我们脚本中存在类A时,我们再从其他脚本引入一个类A时带上完整的路径来区分不同的类A。
PHP引入命名空间的意义
上面青锋建站提到命名空间解决了类冲突的问题。可能有人会问,一般不会存在类冲突,我也没有遇到类冲突的情况。确实存在这样的情况,就连青锋建站自己的团队写程序也不使用命名空间,但是命名空间主要应用在PHP框架结构中。我们自己写程序一般可以人为的避开名称冲突,但是假如我们写得是框架,而框架在引入其他人开发的模块或使用其他人开发的工具时就会遇到大量的类名冲突,这时命名空间就很有意义;再者,假如我们想多个模块存在index类,就会产生名称冲突,PHP命名空间也为我们解决了使用重名的类的问题。官方说法
PHP官方命名空间讲解https://www.php.net/manual/zh/language.namespaces.phpPHP版本要求(PHP 5 >= 5.3.0, PHP 7)
什么是命名空间?从广义上来说,命名空间是一种封装事物的方法。在很多地方都可以见到这种抽象概念。例如,在操作系统中目录用来将相关文件分组,对于目录中的文件来说,它就扮演了命名空间的角色。具体举个例子,文件 foo.txt 可以同时在目录/home/greg 和 /home/other 中存在,但在同一个目录中不能存在两个 foo.txt 文件。另外,在目录 /home/greg 外访问 foo.txt 文件时,我们必须将目录名以及目录分隔符放在文件名之前得到 /home/greg/foo.txt。这个原理应用到程序设计领域就是命名空间的概念。在PHP中,命名空间用来解决在编写类库或应用程序时创建可重用的代码如类或函数时碰到的两类问题:
1、用户编写的代码与PHP内部的类/函数/常量或第三方类/函数/常量之间的名字冲突。
2、为很长的标识符名称(通常是为了缓解第一类问题而定义的)创建一个别名(或简短)的名称,提高源代码的可读性。
命名空间的定义
虽然任意合法的PHP代码都可以包含在命名空间中,但只有以下类型的代码受命名空间的影响,它们是:类(包括抽象类和traits)、接口、函数和常量。
命名空间通过关键字namespace 来声明。如果一个文件中包含命名空间,它必须在其它所有代码之前声明命名空间,除了一个以外:declare关键字。在声明命名空间之前唯一合法的代码是用于定义源文件编码方式的 declare 语句。另外,所有非 PHP 代码包括空白符都不能出现在命名空间的声明之前:
另外,与PHP其它的语言特征不同,同一个命名空间可以定义在多个文件中,即允许将同一个命名空间的内容分割存放在不同的文件中。
1、非限定名称,或不包含前缀的类名称,例如 $a=new foo(); 或 foo::staticmethod();。如果当前命名空间是 currentnamespace,foo 将被解析为 currentnamespaceoo。如果使用 foo 的代码是全局的,不包含在任何命名空间中的代码,则 foo 会被解析为foo。 警告:如果命名空间中的函数或常量未定义,则该非限定的函数名称或常量名称会被解析为全局函数名称或常量名称。详情参见 使用命名空间:后备全局函数名称/常量名称。
2、限定名称,或包含前缀的名称,例如 $a = new subnamespaceoo(); 或 subnamespaceoo::staticmethod();。如果当前的命名空间是 currentnamespace,则 foo 会被解析为 currentnamespacesubnamespaceoo。如果使用 foo 的代码是全局的,不包含在任何命名空间中的代码,foo 会被解析为subnamespaceoo。
3、完全限定名称,或包含了全局前缀操作符的名称,例如, $a = new currentnamespaceoo(); 或 currentnamespaceoo::staticmethod();。在这种情况下,foo 总是被解析为代码中的文字名(literal name)currentnamespaceoo。
另外,与PHP其它的语言特征不同,同一个命名空间可以定义在多个文件中,即允许将同一个命名空间的内容分割存放在不同的文件中。
使用命名空间
PHP 命名空间中的元素使用同样的原理。例如,类名可以通过三种方式引用:1、非限定名称,或不包含前缀的类名称,例如 $a=new foo(); 或 foo::staticmethod();。如果当前命名空间是 currentnamespace,foo 将被解析为 currentnamespaceoo。如果使用 foo 的代码是全局的,不包含在任何命名空间中的代码,则 foo 会被解析为foo。 警告:如果命名空间中的函数或常量未定义,则该非限定的函数名称或常量名称会被解析为全局函数名称或常量名称。详情参见 使用命名空间:后备全局函数名称/常量名称。
2、限定名称,或包含前缀的名称,例如 $a = new subnamespaceoo(); 或 subnamespaceoo::staticmethod();。如果当前的命名空间是 currentnamespace,则 foo 会被解析为 currentnamespacesubnamespaceoo。如果使用 foo 的代码是全局的,不包含在任何命名空间中的代码,foo 会被解析为subnamespaceoo。
3、完全限定名称,或包含了全局前缀操作符的名称,例如, $a = new currentnamespaceoo(); 或 currentnamespaceoo::staticmethod();。在这种情况下,foo 总是被解析为代码中的文字名(literal name)currentnamespaceoo。
下面是一个使用这三种方式的实例:
file1.php
<?php
namespace FooBarsubnamespace;
const FOO = 1;
function foo() {}
class foo
{
static function staticmethod() {}
}
?>
namespace FooBarsubnamespace;
const FOO = 1;
function foo() {}
class foo
{
static function staticmethod() {}
}
?>
file2.php
<?php
namespace FooBar;
include 'file1.php';
const FOO = 2;
function foo() {}
class foo
{
static function staticmethod() {}
}
/* 非限定名称 */
foo(); // 解析为 FooBaroo resolves to function FooBaroo
foo::staticmethod(); // 解析为类 FooBaroo的静态方法staticmethod。resolves to class FooBaroo, method staticmethod
echo FOO; // resolves to constant FooBarFOO
/* 限定名称 */
subnamespaceoo(); // 解析为函数 FooBarsubnamespaceoo
subnamespaceoo::staticmethod(); // 解析为类 FooBarsubnamespaceoo,
// 以及类的方法 staticmethod
echo subnamespaceFOO; // 解析为常量 FooBarsubnamespaceFOO
/* 完全限定名称 */
FooBaroo(); // 解析为函数 FooBaroo
FooBaroo::staticmethod(); // 解析为类 FooBaroo, 以及类的方法 staticmethod
echo FooBarFOO; // 解析为常量 FooBarFOO
?>
namespace FooBar;
include 'file1.php';
const FOO = 2;
function foo() {}
class foo
{
static function staticmethod() {}
}
/* 非限定名称 */
foo(); // 解析为 FooBaroo resolves to function FooBaroo
foo::staticmethod(); // 解析为类 FooBaroo的静态方法staticmethod。resolves to class FooBaroo, method staticmethod
echo FOO; // resolves to constant FooBarFOO
/* 限定名称 */
subnamespaceoo(); // 解析为函数 FooBarsubnamespaceoo
subnamespaceoo::staticmethod(); // 解析为类 FooBarsubnamespaceoo,
// 以及类的方法 staticmethod
echo subnamespaceFOO; // 解析为常量 FooBarsubnamespaceFOO
/* 完全限定名称 */
FooBaroo(); // 解析为函数 FooBaroo
FooBaroo::staticmethod(); // 解析为类 FooBaroo, 以及类的方法 staticmethod
echo FooBarFOO; // 解析为常量 FooBarFOO
?>
使用命名空间:别名/导入
允许通过别名引用或导入外部的完全限定名称,是命名空间的一个重要特征。这有点类似于在类 unix 文件系统中可以创建对其它的文件或目录的符号连接。所有支持命名空间的PHP版本支持三种别名或导入方式:为类名称使用别名、为接口使用别名或为命名空间名称使用别名。PHP 5.6开始允许导入函数或常量或者为它们设置别名。
在PHP中,别名是通过操作符 use 来实现的。
所有可能的五种导入名称空间的例子
<?php
namespace foo;
use MyFullClassname as Another;
// 下面的例子与 use MyFullNSname as NSname 相同
use MyFullNSname;
// 导入一个全局类
use ArrayObject;
// importing a function (PHP 5.6+)
use function MyFullunctionName;
// aliasing a function (PHP 5.6+)
use function MyFullunctionName as func;
// importing a constant (PHP 5.6+)
use const MyFullCONSTANT;
$obj = new namespaceAnother; // 实例化 fooAnother 对象
$obj = new Another; // 实例化 MyFullClassname 对象
NSnamesubnsunc(); // 调用函数 MyFullNSnamesubnsunc
$a = new ArrayObject(array(1)); // 实例化 ArrayObject 对象
// 如果不使用 "use ArrayObject" ,则实例化一个 fooArrayObject 对象
func(); // calls function MyFullunctionName
echo CONSTANT; // echoes the value of MyFullCONSTANT
?>
注意对命名空间中的名称(包含命名空间分隔符的完全限定名称如 FooBar以及相对的不包含命名空间分隔符的全局名称如 FooBar)来说,前导的反斜杠是不必要的也不推荐的,因为导入的名称必须是完全限定的,不会根据当前的命名空间作相对解析。namespace foo;
use MyFullClassname as Another;
// 下面的例子与 use MyFullNSname as NSname 相同
use MyFullNSname;
// 导入一个全局类
use ArrayObject;
// importing a function (PHP 5.6+)
use function MyFullunctionName;
// aliasing a function (PHP 5.6+)
use function MyFullunctionName as func;
// importing a constant (PHP 5.6+)
use const MyFullCONSTANT;
$obj = new namespaceAnother; // 实例化 fooAnother 对象
$obj = new Another; // 实例化 MyFullClassname 对象
NSnamesubnsunc(); // 调用函数 MyFullNSnamesubnsunc
$a = new ArrayObject(array(1)); // 实例化 ArrayObject 对象
// 如果不使用 "use ArrayObject" ,则实例化一个 fooArrayObject 对象
func(); // calls function MyFullunctionName
echo CONSTANT; // echoes the value of MyFullCONSTANT
?>
PHP使用多个use语句
Example #2 通过use操作符导入/使用别名,一行中包含多个use语句
<?php
use MyFullClassname as Another, MyFullNSname;
$obj = new Another; // 实例化 MyFullClassname 对象
NSnamesubnsunc(); // 调用函数 MyFullNSnamesubnsunc
?>
PHP名称空间解析规则
1、对完全限定名称的函数,类和常量的调用在编译时解析。例如 new AB 解析为类 AB。
2、所有的非限定名称和限定名称(非完全限定名称)根据当前的导入规则在编译时进行转换。例如,如果命名空间 ABC 被导入为 C,那么对 CDe() 的调用就会被转换为 ABCDe()。
3、在命名空间内部,所有的没有根据导入规则转换的限定名称均会在其前面加上当前的命名空间名称。例如,在命名空间 AB 内部调用 CDe(),则 CDe() 会被转换为 ABCDe() 。
4、非限定类名根据当前的导入规则在编译时转换(用全名代替短的导入名称)。例如,如果命名空间 ABC 导入为C,则 new C() 被转换为 new ABC() 。
5、在命名空间内部(例如AB),对非限定名称的函数调用是在运行时解析的。例如对函数 foo() 的调用是这样解析的:
在当前命名空间中查找名为 ABoo() 的函数
尝试查找并调用 全局(global) 空间中的函数 foo()。
6、在命名空间(例如AB)内部对非限定名称或限定名称类(非完全限定名称)的调用是在运行时解析的。下面是调用 new C() 及 new DE() 的解析过程: new C()的解析:
在当前命名空间中查找ABC类。
尝试自动装载类ABC。
new DE()的解析:
在类名称前面加上当前命名空间名称变成:ABDE,然后查找该类。
尝试自动装载类 ABDE。
为了引用全局命名空间中的全局类,必须使用完全限定名称 new C()。
以上就是青锋建站给大家整理的PHP命名空间的简单使用与概念详解。青锋建站,提供专业的高品质网站制作服务,包括PHP网站建设,SEO,网络营销,开发面向CMS建站开发,PHP原生开发,采用PHP框架开发。
use MyFullClassname as Another, MyFullNSname;
$obj = new Another; // 实例化 MyFullClassname 对象
NSnamesubnsunc(); // 调用函数 MyFullNSnamesubnsunc
?>
PHP名称空间解析规则
1、对完全限定名称的函数,类和常量的调用在编译时解析。例如 new AB 解析为类 AB。
2、所有的非限定名称和限定名称(非完全限定名称)根据当前的导入规则在编译时进行转换。例如,如果命名空间 ABC 被导入为 C,那么对 CDe() 的调用就会被转换为 ABCDe()。
3、在命名空间内部,所有的没有根据导入规则转换的限定名称均会在其前面加上当前的命名空间名称。例如,在命名空间 AB 内部调用 CDe(),则 CDe() 会被转换为 ABCDe() 。
4、非限定类名根据当前的导入规则在编译时转换(用全名代替短的导入名称)。例如,如果命名空间 ABC 导入为C,则 new C() 被转换为 new ABC() 。
5、在命名空间内部(例如AB),对非限定名称的函数调用是在运行时解析的。例如对函数 foo() 的调用是这样解析的:
在当前命名空间中查找名为 ABoo() 的函数
尝试查找并调用 全局(global) 空间中的函数 foo()。
6、在命名空间(例如AB)内部对非限定名称或限定名称类(非完全限定名称)的调用是在运行时解析的。下面是调用 new C() 及 new DE() 的解析过程: new C()的解析:
在当前命名空间中查找ABC类。
尝试自动装载类ABC。
new DE()的解析:
在类名称前面加上当前命名空间名称变成:ABDE,然后查找该类。
尝试自动装载类 ABDE。
为了引用全局命名空间中的全局类,必须使用完全限定名称 new C()。
以上就是青锋建站给大家整理的PHP命名空间的简单使用与概念详解。青锋建站,提供专业的高品质网站制作服务,包括PHP网站建设,SEO,网络营销,开发面向CMS建站开发,PHP原生开发,采用PHP框架开发。
转载请注明来源网址:青锋建站-http://www.sjzphp.com/webdis/php_namespace_870.html