命名空间一个最明确的目的就是解决重名问题,PHP中不允许两个函数或者类出现相同的名字,否则会产生一个致命的错误。这种情况下只要避免命名重复就可以解决,最常见的一种做法是约定一个前缀,但是方法名变的很长了,而且仍可能重名。为了避免这一情况,启用命名空间,虽然会增加一了一点使用麻烦度。
文档copy了段:什么是命名空间?从广义上来说,命名空间是一种封装事物的方法。在很多地方都可以见到这种抽象概念。例如,在操作系统中目录用来将相关文件分组,对于目录中的文件来说,它就扮演了命名空间的角色。具体举个例子,文件foo.txt 可以同时在目录/home/greg 和 /home/other 中存在,但在同一个目录中不能存在两个 foo.txt 文件。另外,在目录/home/greg 外访问 foo.txt 文件时,我们必须将目录名以及目录分隔符放在文件名之前得到 /home/greg/foo.txt。这个原理应用到程序设计领域就是命名空间的概念。
在PHP中,命名空间用来解决在编写类库或应用程序时创建可重用的代码如类或函数时碰到的两类问题:
- 用户编写的代码与PHP内部的类/函数/常量或第三方类/函数/常量之间的名字冲突。
- 为很长的标识符名称(通常是为了缓解第一类问题而定义的)创建一个别名(或简短)的名称,提高源代码的可读性。
本文提到的常量:PHP5.3开始const关键字可以用在类的外部。const和define都是用来声明常量的(它们的区别不详述),但是在命名空间里,define的作用是全局的,而const则作用于当前空间。我在文中提到的常量是指使用const声明的常量。
直接看例子慢慢体会理解。
Demo.php内容如下
namespace DemoNameSpace; class Demo { private $mysqlHandle; public function __construct() { echo 'This is namespace of PHP demo ,The Demo magic constant "__NAMESPACE__" is '.__NAMESPACE__; } }
index.php代码
include 'Demo.php'; use DemoNameSpaceDemo; $DemoObj = new Demo();
输出结果1:“This is namespace of PHP demo ,The Demo magic constant "__NAMESPACE__" is DemoNameSpace”
以上示例的说明:Demo.php中有一个_NAMESPACE__魔法常量;“它包含当前命名空间名称的字符串。在全局的,不包括在任何命名空间中的代码,它包含一个空的字符串。”
接着做示例:
Demo.php不做变动,改动index.php文件,如下:
include 'Demo.php'; $Demo = new Demo();
输出结果2:“Fatal error: Class 'Demo' not found in F:JJserverdemoindex.php on line 4”
这个是常见的“致命错误”信息了。按照常规的php编程思路,这里的输出应该是跟“输出结果1”一致的,但这里它偏要来个致命错误,这下要抓狂了吧?~
行,先把抓狂的麻烦解决,去掉(或注释掉)Demo.php文件中的:“namespace DemoNameSpace;”这个语句,就正常了。这是咱们平常写class以及调用class最常见的书写方法,就不再解释这种不使用namespace的情况了。
对比使用namespace与不使用namespace的两种输出情况,并加入namespace的定义理解后,上述出现致命错误的情况就很好理解了。在Demo.php中定义了一个namespace,也就是命名空间后,接着定义了Demo类,然后这个Demo类就被归并到了DemoNameSpace命名空间中去了,那么要调用这个Demo类时,自然要先调用这个DemoNameSpace命名空间了,也就是在index.php文件中使用“useDemoNameSpaceDemo”语句了。
再看个复杂点的例子
Demo.php文件代码
namespace DemoNameSpace; class Demo { private $mysqlHandle; public function __construct() { echo 'This is namespace of PHP demo ,The Demo magic constant "__NAMESPACE__" is '.__NAMESPACE__; } } namespace DemoNameSpace1; const constDefine = 'JJonline1'; class Demo { private $mysql; const constDefine = 'JJonline2'; public function __construct() { echo 'The const constant outside class is: '.constDefine; echo '===cut-off rule of god!!!!==='; echo 'The const constant inside class is: '.self::constDefine; } }
index.php文件代码
include 'Demo.php'; use DemoNameSpace1Demo as Test; $Demo = new Test(); echo '||||'.DemoNameSpace1constDefine;
输出结果3:“The const constant outside class is: JJonline1===cut-off rule of god!!!!===The const constant inside class is: JJonline2||||JJonline1”
这个结果在没有命名空间的时候,就直接报诸如“Fatal error: Cannot redeclare class Demo”的致命错误了。但运行没有报错,这也就是php5.3以后引入的命名空间的好处了,就诸如本文开头引用的官方解释中以不同目录下的相同文件名的文件可以存在一样是一个道理了。Demo.php文件中,定义的第一个名称叫做Demo的class类被归并到了DemoNameSpace的命名空间,而定义的第二个名称叫做Demo的class被归并到了DemoNameSpace1的命名空间,故而并不会出现不能重复定义某一个类的致命错误。以上的书写方法是要尽量避免的,因为类外部const常量名与类内部const常量名是一样的,很容易混淆,这里这样书写的目的就是看看不同位置申明的const常量,在调用时的情况;输出结果3已经很明显了,就不再多墨迹解释了。
Demo.php中DemoNameSpace1命名空间下还将const常量constDefine提出,拿到了定义class之外,这又要抓狂了,因为之前的知识是define定义全局常量,const定义class内部常量;这儿却将const拿出来玩了...具体就不再讲解了,Demo.php文件代码以及运行后的结果已经很明确的表明了相关知识。class内部定义的const只能在class的内部调用,采用self::constName形式,而class内部调用命名空间下、class外的const常量,则可以直接使用诸如define定义的常量一样使用。当需要使用该命名空间下、class外定义的const常量时,就使用类似路径形式的方式调用(index.php文件中的输出)。
该例子还有一点说明,就是在index.php中使用了use as语句,看index.php的代码,意义一目了然,new的一个class名称叫Test,但Test这个类并没有在Demo.php中定义,却没有出错,这就在于了use as语句了,具体意义不再解释。
通过上述的了解,namespace关键字可以将实现各种功能的class通过指定不同的命名空间分门别类存放,而且不同命名空间下的class可以同名;另外const常量定义也可以提出到class外部,当然也会有作用范围这么一个“内涵”~
当前脚本文件的第一个命名空间前面不能有任何代码,例如如下代码就是会报致命错误的:
define("GREETING","Hello world!"); namespace DemoNameSpace; class Demo { private $mysqlHandle; public function __construct() { echo 'This is namespace of PHP demo ,The Demo magic constant "__NAMESPACE__" is '.__NAMESPACE__; } } $Demo = new Demo();
运行上述代码,会出现致命错误:“Fatal error: Namespace declaration statement has to be the very first statement in xxxx”
命名空间下直接new该命名空间中的class名称,可以省略掉use语法,这是php按脚本书写顺序执行导致的。例如如下代码是可以运行的
namespace DemoTest; class Demo { public function __construct() { echo 'this is a test script'; } } namespace DemoNameSpace; class Demo { private $mysqlHandle; public function __construct() { echo 'This is namespace of PHP demo ,The Demo magic constant "__NAMESPACE__" is '.__NAMESPACE__; } } $Demo = new Demo();
运行结果4:“This is namespace of PHP demo ,The Demo magic constant "__NAMESPACE__" is DemoNameSpace”
这个结果表明,同一脚本下new一个没有指定use哪个命名空间时,会顺着该脚本,使用最靠近new语句之前的一个命名空间中的class
本文转载自晶晶的博客。。。。