A-A+

PHP命令空间namespace及use的用法小结

2021年08月22日 我爱编程 暂无评论

命名空间一个最明确的目的就是解决重名问题,PHP中不允许两个函数或者类出现相同的名字,否则会产生一个致命的错误。这种情况下只要避免命名重复就可以解决,最常见的一种做法是约定一个前缀。

使用namespace的目的:

团队合作项目时,避免与团队其它成员新建的类发生冲突;个人负责项目时,避免前后新建的类发生冲突;

据个人理解,用到所需要的类时,需要先require或include引入,所以会发生类重定义的错误的前提是:两个相同命名的类都有被引入。目前有些php框架会自动加载(即include)所有新建的model类,所以为了避免你新建的model类和项目框架原生的核心类发生重名冲突,采用了namespace。(想了想,与团队成员新建的类产生冲突应该通过沟通避免,即使事发后也应该重新调整类名即时维护,避免后期造成因为对类的理解混淆而带来维护上的复杂度提高)

结合使用方法来进一步理解它的使用目的吧。

namespace的使用方法:

为了进行测试,我将创建3个文件:1.php和name.php(此文件用来执行测试),后面将不再说明,请自行注意代码的变化。

1.namespace后命名的定义不区分大小写

namespace one;

namespace One;

namespace ONE;

如上写法都可以,选择一种作为自己的规范即可,(后面代码我采用第一种进行测试哈)

2. 没有定义命名空间,就理解为使用顶级命名空间,new类时,可以在类前加上反斜杠\,也可以不加。

  1. //1.php 
  2. class Person{
  3.  function __construct(){
  4.   echo 'I am one!';
  5.  }
  6. }
  7. //name.php
  8. require_once './1.php';
  9. new Person(); //输出 I am one!;
  10. new \Person(); //输出 I am one!;

3. new类时,带上命名空间时,之间一定用反斜杠字符,而不是顺斜杠。

记忆方法:按找%中斜杠的顺序理解为顺斜杠。(有时说反斜杠,自己都不知道是哪种方向,以前按自左向右上升方向这种方向记忆,现在感觉这个太不靠谱了)

  1. //name.php
  2. require_once './1.php';
  3. new /Person(); // 代码报错:Parse error: syntax error, unexpected '/'

4.类在指定命名空间下, new类时,一定要带上指定的命名空间。

没有带上指定的命名空间,按照第2点,php就会从顶级命名空间里找这个类。切记:这里不能按照顶级命名空间包含一切其它的命名空间来理解。而应该将顶级命名空间完完全全与其他命名空间区分开。

  1. //1.php 
  2. namespace one;
  3. class Person{
  4.  function __construct(){
  5.   echo 'I am one!';
  6.  }
  7. }
  8. //name.php
  9. require_once './1.php';
  10. new \one\Person(); //输出 I am one!;
  11. new \Person(); //代码报错:Fatal error: Class 'Person' not found

可以举个这个通俗例子来理解:带上指定命名空间代表某人的苹果(在他手里),顶级命名空间代表苹果箱里的苹果(在箱子里)。现在要找某人的苹果,就将某人的命名空间带上,否则就会从箱子里找某人的苹果,结果当然是找不到。

5.命名空间声明后的代码便属于这个命名空间,即使有include或require也不影响(重点是对后半句的理解,具体看代码)。

  1. //1.php 
  2. namespace one;
  3. class Person{
  4.  function __construct(){
  5.   echo 'I am one!';
  6.  }
  7. }
  8. //name.php
  9. namespace test;
  10. require './1.php';
  11. new \one\Person(); //输出 I am one!;
  12. new Person(); //这里结果会是什么呢,猜猜看

最后一行结果报错:

Fatal error: Class 'test\Person' not found

首先,这里与第2点比较一下:

第2点,我说,没有命名空间时,new类时,有没有反斜杠意义一样。

在这里,有了命名空间,有和没有反斜杠的意义就不一样了。

最后一行换成

new \Person();

结果报错:

Fatal error: Class 'Person' not found

接着,就说说当前这点。

我们可以发现,最后一行代码对应的命名空间为test,并没有受到require文件里的命名空间的影响。

进一步加强验证,我修改了name.php文件如下:

  1. //name.php
  2. namespace test;
  3. require './1.php';
  4. class Person{
  5.  function __construct(){
  6.   echo 'I am test!';
  7.  }
  8. }
  9. new \one\Person(); //输出 I am one!;
  10. new Person(); //这里结果会是什么,自己猜猜看

最后,这个例子刷新了我对require的认识了。

按照我以前对require的理解:PHP 程序在执行前,就会先读入 require 所指定引入的文件,使它变成 PHP 程序网页的一部分。所以我常常就简单的理解为替换,只不过把抽离出来的代码再放回原处而已。然后我试了将1.php文件内容放到name.php里:

  1. //name.php
  2. namespace test;
  3. namespace one;
  4. class Person{
  5.  function __construct(){
  6.   echo 'I am one!';
  7.  }
  8. }
  9. class Person{
  10.  function __construct(){
  11.   echo 'I am test!';
  12.  }
  13. }

无需new类,该文件就会报错:

Fatal error:  Cannot redeclare class one\Person

看来简单的把require理解为替换,在这里行不通。

6.namespace里不包含类名称,即使存在与类名称同名的部分,那也不代表类。new类时,还是得将这部分带上。

  1. //name.php
  2. namespace test\person;
  3. class Person{
  4.  function __construct(){
  5.   echo 'I am test!';
  6.  }
  7. }new \test\person\Person();  //命名空间里person无法代表类名

不过这样纯粹是画蛇添足,直接干脆点,namespace里不要带类名称就好。

7.一个php文件中可以存在多个命名空间,第一个命名空间前不能有任何代码。

只说第一个命名空间前不能有任何代码,之后的命名空间之前可以有代码。这个自行测试即可。

  1. //name.php
  2. namespace test;
  3. echo 'zhai14';
  4. namespace zhai;
  5. require './1.php';

php命名空间namespace告一段落了,接下来说说use的用途。

使用use的目的:

在命名空间字符串过长时,使用use可以相应的缩短命名空间。

use的使用方法:

1.new类时,最前面无需用反斜杠,此外,use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。

  1. //name.php
  2. namespace animal\dog;
  3. class Life{
  4.  function __construct(){
  5.   echo 'dog life!';
  6.  }
  7. }
  8. namespace animal\cat;
  9. class Life{
  10.  function __construct(){
  11.   echo 'cat life!';
  12.  }
  13. }
  14. new Life(); //按照代码执行顺序,这里默认animal\cat这个命名空间
  15. new \animal\dog\Life(); //A
  16. use animal\dog; //a
  17. new dog\Life(); //B
  18. use animal\dog as d; //b
  19. new d\Life();

通过A、B行代码比较,需要注意:

使用use后,new类时,最前面没有反斜杠。

没使用use时,命名空间最前面有反斜杠

通过a、b行代码比较,可以理解:

use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容,如上的:

use animal\dog;

相当于

use animal\dog as dog;

2.namespace后面不建议加类名,但use后可以。

  1. //name.php
  2. namespace animal\dog;
  3. class Life{
  4.  function __construct(){
  5.   echo 'dog life!';
  6.  }
  7. }
  8. namespace animal\cat;
  9. class Life{
  10.  function __construct(){
  11.   echo 'cat life!';
  12.  }
  13. }
  14. use animal\dog\Life as dog;
  15. new dog();

如上所示,use后加上类名后,就相当于把类改了个名称:由Life改为dog了。

上面不用as dog就会报错:

Fatal error:  Cannot use animal\dog\Life as Life because the name is already in use

因为cat下也有个一样名称的Life类。

可以理解为,使用use后,这个昵称对应的类只能归当前命名空间占有,其它命名空下不允许存在该类。

  1. //name.php
  2. namespace animal\dog;
  3. class Life{
  4.  function __construct(){
  5.   echo 'dog life!';
  6.  }
  7. }
  8. class Dog{
  9.  function __construct(){
  10.   echo 'dog in dog!';
  11.  }
  12. }
  13. namespace animal\cat;
  14. // class Dog{
  15. // function __construct(){
  16. //  echo 'dog in cat!';
  17. //  }
  18. // }
  19. class Life{
  20.  function __construct(){
  21.   echo 'cat life!';
  22.  }
  23. }
  24. use animal\dog;
  25. new dog\Dog();

如上,使用了

use animal\dog;

cat

通过上面代码,我想使用use的目的效果(缩短命名空间名称)就很明显了。

简单总结一下:

namespace就是划分领域的作用,代表这些东西是属于某个命名空间下的。

use就是起小名的作用,不论写起来还是说起来都可以省不少事儿。

标签:

给我留言

Copyright © 四季博客 保留所有权利.   Theme  Ality

用户登录