PHP中的命名空间详细介绍

   这篇文章主要介绍了PHP中的命名空间详细介绍,本文讲解了命名空间(namespace)的概念、正在使用命名空间、定义命名空间、子命名空间、从命名空间中调用代码等内容,需要的朋友可以参考下

  概述

  PHP对于命名空间的支持,经历了一段艰难的旅程。幸运的是,PHP从5.3开始引入了命名空间。自从PHP引入了命名空间,PHP代码的适用结构也得到了大大的改善。许多编程语言早就有了命名空间的概念,相对于其他语言来说,PHP对于命名空间的支持,稍微有点晚了。不管如何,每一种新特性的引入都有其目的,和其他语言一样,PHP引入命名空间也主要是为了解决名字冲突的问题。

  命名空间(namespace)的概念

   代码如下:

  当在字符串中使用命名空间名字的时候,一定不要忘了转义

  可以将命名空间想象成一个抽屉,你可以在抽屉里放入铅笔、尺子、A4纸等,这些都是你自己的私有物品。在你的抽屉下面是别人的抽屉,别人也可以在抽屉里放入相同的物品。为了不拿错物品,你们决定在自己的抽屉上贴上标签,这样就可以清晰的看到某个物品是属于谁的了。

  之前,开发者必须在类、函数和常量中添加下划线,用来使自己的代码独立其他于代码库。这相当于所有人都给自己的物品贴上标签之后,一起放入了一个更大的抽屉里。尽管这也是一种组织代码的方式,但是这种方式是非常低效的。

  命名空间的到来就是为了解决这个问题。我们可以在不同的命名空间里声明相同的函数、类和常量,而不会造成名字上的冲突。本质上,命名空间无非是一种分等级标记PHP代码的方式。

  正在使用命名空间

  有一点需要注意的是,我们正在间接的使用命名空间。从PHP 5.3开始,所有在非用户定义的命名空间中的声明(类、函数、常量),都默认的属于全局命名空间。

  全局命名空间中包含了所有PHP内部的定义,如echo()、mysqli_connect()和Exception类。由于全局命名空间并没有独立的标识名,所以它经常被成为全局空间(global space)。

  定义命名空间

  命名空间的定义必须是PHP文件的第一条语句。唯一允许在定义命名空间之前使用的语句是declare语句。

  定义命名空间很简单,只需要使用关键字namespace即可。命名空间的名字需要遵循PHP文件中其他标识符的命名规则。

  下面是定义一个命名空间的示例:

   代码如下:

  namespace MyNamespace{

  class Test{

  }

  }

  如果想定义一个属于全局空间的代码块,也是使用namespace关键字,但是后面不加命名空间的名字,如下:

  代码如下:

  namespace {

  class Test{

  }

  }

  我们甚至可以在一个文件中定义多个命名空间,如下:

   代码如下:

  

  namespace MyNamespace {

  }

  namespace MySecondNamespace {

  }

  namespace {

  }

  我们也可以将一个命名空间分散在不同的文件中,文件包含的处理程序会自动合并他们。因此,限制大量的命名空间在同一个文件中定义是一个很好的编程实践,就像我们通常单独为每个类定义一个单独的文件一样。

  复制代码 代码如下:

  有一点需要注意的是,包含命名空间代码块的{是可选的,可以用也可以不用。事实上,只要我们坚持在一个文件中只定义一个命名空间,那么我们就可以完全省略{,这样也可以使我们的代码看起来更加简洁。

  子命名空间

  命名空间可以遵循一个特定的层级,就像我们电脑文件系统中得目录一样。子命名空间对于将一个项目结构化尤其特别有用。例如,你的项目需要访问数据库,你可能会想将所有数据库相关的代码(如数据库异常处理等)放在同一个子目录下。

  为了保持灵活性,将子命名空间放在子目录中是非常明智的做法。这会使你的代码结构更清晰,而且会使遵循PSR-0标准的autoloaders的使用变得更容易。

  PHP使用反斜线作为命名空间的分隔符,有趣的是,PHP甚至考虑过使用笑脸:)作为命名空间的分隔符。

  子命名空间定义示例:

  代码如下:

  

  namespace MyProjectDatabase

  class Connection {

  }

  可以使用尽可能多的子命名空间:

   代码如下:

  

  namespace MyProjectBlogAuthHandlerSocial;

  class Twitter {

  }

  有一点需要注意的是,PHP并不支持命名空间的嵌套定义,下面的代码会导致一个致命错误:Namespace declarations cannot be nested。

   代码如下:

  

  namespace MyProject {

  namespace Database {

  class Connection { }

  }

  }

  从命名空间中调用代码

  如果你想在不同的命名空间中实例化一个类、调用一个函数或者使用常量,需要使用反斜线。他们可以从三个角度被解析:

  1.未限定的名字

  2.限定的名字

  3.完全限定的名字

  未限定的名字(Unqualified Name)

  这是一个类的名称,函数或常量,但是不包括任何命名的引用。如果命名空间对你来说还比较陌生,那么这就是你熟悉的角度。

   代码如下:

  

  namespace MyProject;

  class MyClass {

  static function static_method()

  {

  echo 'Hello, world!';

  }

  }

  // Unqualified name, resolves to the namespace you are currently in (MyProjectMyClass)

  MyClass:static_method();

  限定的名字(Qualified Name)

  这是我们如何使用子命名空间的方式。示例如下:

  代码如下:

  

  namespace MyProject;

  require 'myproject/database/connection.php';

  // Qualified name, instantiating a class from a sub-namespace of MyProject

  $connection = new DatabaseConnection();

  完全限定的名字(Fully Qualified Name)

  前面所说的使用限定的名字和未限定的名字,都是相对于当前所处的命名空间来说的。以上两种方式仅可以被用来访问当前所处的命名空间和更深层次的子命名空间。

  如果想访问一个在比前命名空间更高的层级,那么就需要使用完全限定的名字—一个绝对路径而不是相对路径。这可以归结为在命名空间的最前面加反斜杠。使用完全限定的名字可以让PHP知道,这次调用是从全局空间开始的,而不是相对于当前所处的命名空间。示例如下:

  代码如下:

  

  namespace MyProjectDatabase;

  require 'myproject/fileaccess/input.php';

  // Trying to access the MyProjectFileAccessInput class

  // This time it will work because we use the fully qualified name, note the leading backslash

  $input = new MyProjectFileAccessInput();

  对于PHP的内部函数来说,我们不必要使用完全限定的名字。在当前所处的命名空间中,调用一个不存在的未限定的名字的类或函数,PHP会搜索全局空间。

  记住了这个规则,我们就可以像下面那样重写PHP的内部函数:

   代码如下:

  

  namespace MyProject;

  var_dump($query); // Overloaded

  var_dump($query); // Internal

  // We want to access the global Exception class

  // The following will not work because there's no class called Exception in the MyProjectDatabase namespace and unqualified class names do not have a fallback to global space

  // throw new Exception('Query failed!');

  // Instead, we use a single backslash to indicate we want to resolve from global space

  throw new Exception('ailed!');

  function var_dump() {

  echo 'Overloaded global var_dump()!
';

  }

  动态调用

  PHP是一门动态语言,也可以将PHP的这种特性用来调用命名空间。这在本质上与实例化一个变量类和包含一个变量文件是相同的。在字符串中,PHP使用的命名空间分隔符()也是一个元字符,因此需要转义。

   代码如下:

  

  namespace OtherProject;

  $project_name = 'MyProject';

  $package_name = 'Database';

  $class_name = 'Connection';

  // Include a variable file

  require strtolower($project_name . '/'. $package_name . '/' . $class_name) . '.php';

  // Name of a variable class in a variable namespace. Note how the backslash is escaped to use it properly

  $fully_qualified_name = $project_name . '' . $package_name . '' . $class_name;

  $connection = new $fully_qualified_name();

  namespace关键字

  关键字namespace不仅仅可以用来定义一个命名空间,它也可以用来显示的表示当前命名空间,它此时的作用相当于类中的self关键字。

   代码如下:

  

  namespace MyProject;

  function run()

  {

  echo 'Running from a namespace!';

  }

  // Resolves to MyProjectrun

  run();

  // Explicitly resolves to MyProjectrun

  namespacerun();

  __NAMESPACE__常量

  就像self关键字不能表示当前类的名字一样,namespace关键字也不能用来表示当前命名空间的名字。__NAMESPACE__关键字就是用来解决这个问题的。

  代码如下:

  

  namespace MyProjectDatabase;

  // 'MyProjectDatabase'

  echo __NAMESPACE__;

  这个关键字对于判断当前代码是否从命名空间开始时非常有用,而且也可以用来调试代码。

  导入或别名

  PHP中得命名空间也支持导入,导入也被成为别名。只有类、接口和命名空间可以被导入(别名)。导入是命名空间中一个非常有用和基础的功能。它使我们可以使用外部的代码包,而不用担心名字的冲突。使用use关键字可以实现导入功能。也可以使用as关键字,在导入的时候指定一个别名。

   代码如下:

  use [name of class, interface or namespace] as [optional_custom_alias]

  一个完全限定的名字可以用一个未限定的别名来代替,这样我们就不用在每次使用的时候都使用完全限定的名字,达到简化代码的目的。导入应该在命名空间的最高层或者全局空间中使用,在函数作用域内使用导入功能是非法的语法。

   代码如下:

  

  namespace OtherProject;

  // This holds the MyProjectDatabase namespace with a Connection class in it

  require 'myproject/database/connection.php';

  // If we want to access the database connection of MyProject, we need to use its fully qualified name as we're in a different name space

  $connection = new MyProjectDatabaseConnection();

  // Import the Connection class (it works exactly the same with interfaces)

  use MyProjectDatabaseConnection;

  // Now this works too! Before the Connection class was aliased PHP would not have found an OtherProjectConnection class

  $connection = new Connection();

  // Import the MyProjectDatabase namespace

  use MyProjectDatabase;

  $connection = new DatabaseConnection()

  我们可以通过使用别名来简化上面的代码:

   代码如下:

  

  namespace OtherProject;

  require 'myproject/database/connection.php';

  use MyProjectDatabaseConnection as MyConnection;

  $connection = new MyConnection();

  use MyProjectDatabase as MyDatabase;

  $connection = new MyDatabaseConnection();

  总结

  命名空间是用来避免定义冲突,并且为代码引入了更加灵活和组织的方式。有一点需要注意的时,我们并没有义务去使用命名空间,它是和面向对象结合使用的一种工作方式。但是,如果使用了命名空间,我们的代码可能会达到一种新的层次,逼格也会显得更高吧。

时间: 2024-12-02 17:38:15

PHP中的命名空间详细介绍的相关文章

PHP中的命名空间详细介绍_php技巧

概述 PHP对于命名空间的支持,经历了一段艰难的旅程.幸运的是,PHP从5.3开始引入了命名空间.自从PHP引入了命名空间,PHP代码的适用结构也得到了大大的改善.许多编程语言早就有了命名空间的概念,相对于其他语言来说,PHP对于命名空间的支持,稍微有点晚了.不管如何,每一种新特性的引入都有其目的,和其他语言一样,PHP引入命名空间也主要是为了解决名字冲突的问题. 命名空间(namespace)的概念 复制代码 代码如下: 当在字符串中使用命名空间名字的时候,一定不要忘了转义\ 可以将命名空间想

Android Service中方法使用详细介绍

  Android Service中方法使用详细介绍 在Android中,Activity主要负责前台页面的展示,Service主要负责需要长期运行的任务.例如,一个从service播放音乐的音乐播放器,应被设置为前台运行,因为用户会明确地注意它的运行.在状态栏中的通知可能会显示当前的歌曲并且允许用户启动一个activity来与音乐播放器交互. Service的两种实现形式 1.非绑定 通过调用应用程序组件(例如Activity)的startService()方法来启动一个服务.一旦启动,服务就

Android Service中方法使用详细介绍_Android

 service作为四大组件值得我们的更多的关注 在Android中,Activity主要负责前台页面的展示,Service主要负责需要长期运行的任务.例如,一个从service播放音乐的音乐播放器,应被设置为前台运行,因为用户会明确地注意它的运行.在状态栏中的通知可能会显示当前的歌曲并且允许用户启动一个activity来与音乐播放器交互. Service的两种实现形式 1.非绑定 通过调用应用程序组件(例如Activity)的startService()方法来启动一个服务.一旦启动,服务就会在

Android 中的注解详细介绍_Android

注解是我们经常接触的技术,Java有注解,Android也有注解,本文将试图介绍Android中的注解,以及ButterKnife和Otto这些基于注解的库的一些工作原理. 归纳而言,Android中的注解大概有以下好处 提高我们的开发效率 更早的发现程序的问题或者错误 更好的增加代码的描述能力 更加利于我们的一些规范约束 提供解决问题的更优解 准备工作 默认情况下,Android中的注解包并没有包括在framework中,它独立成一个单独的包,通常我们需要引入这个包. dependencies

docker中安装quagga详细介绍_docker

openstack中的虚拟路由器项目占用资源太多,需要将虚拟路由器迁移到Docker中,觉得首先要解决几个问题.      1.如何集成docker到openstack中,这个问题openstack官方给了三种方案,基于nova,heat,和单独的容器项目      2.集成docker后的容器管理编排,决定采用K8S      3.是否可以将quagga装到docker中,打包成quagga镜像以供后边二次开发自动配置程序. 今天尝试在docker中安装quagga,并自动开启zebra,ri

JavaScript中变量作用域详细介绍

  以下变量具有全局作用域:     1.所有定义在最外层的变量(非函数体内部)具有全局作用域.     2.未定义直接赋值的变量,系统会把它声明为全局作用域.     3.所有window对象的属性具有全局作用域. 以下变量具有函数作用域   1. 在函数体内部用var定义的变量,这里要注意一点,只要是在函数里定义的变量,就算是在最       后一句定义,该变量也拥有整个函数的作用域.但是它的赋值是等到运行到那一句代码以         后才赋值的!!!   特别应该说明的一点是,作用域是层

php中session用法详细介绍

在您把用户信息存储到 PHP session 中之前,首先必须启动会话. 注释:session_start() 函数必须位于 <html> 标签之前:  代码如下 复制代码   <?php session_start(); ?> <html> <body> </body> </html>上   面的代码会向服务器注册用户的会话,以便您可以开始保存用户信息,同时会为用户会话分配一个 UID. 存储 Session 变量 存储和取回 se

Android中的WebView详细介绍_Android

Android中WebView的详细解释: 1. 概念: WebView(网络视图)能加载显示网页,可以将其视为一个浏览器.它使用了WebKit渲染引擎加载显示网页. 2. 使用方法: (1).实例化WebView组件: A.在Activity中实例化WebView组件.eg: 复制代码 代码如下:    WebView webView = new WebView(this); B.调用WebView的loadUrl()方法,设置WevView要显示的网页.eg: 复制代码 代码如下: 互联网用

Android中的WebView详细介绍

Android中WebView的详细解释: 1. 概念: WebView(网络视图)能加载显示网页,可以将其视为一个浏览器.它使用了WebKit渲染引擎加载显示网页. 2. 使用方法: (1).实例化WebView组件: A.在Activity中实例化WebView组件.eg: 复制代码 代码如下:    WebView webView = new WebView(this); B.调用WebView的loadUrl()方法,设置WevView要显示的网页.eg: 复制代码 代码如下: 互联网用