PHP中的命名空间详细介绍


Posted in PHP onJuly 02, 2015

概述

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{

  

 }

}

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

namespace MyNamespace {

}

 

namespace MySecondNamespace {

}

 

namespace {

}

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

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

子命名空间

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

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

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

子命名空间定义示例:

<?php 

namespace MyProject\Database

 

class Connection {

  }

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

namespace MyProject\Blog\Auth\Handler\Social;

 

class Twitter {

}

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

<?php

namespace MyProject {

    namespace Database {

        class Connection { }

    }

}

从命名空间中调用代码

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

1.未限定的名字
2.限定的名字
3.完全限定的名字

未限定的名字(Unqualified Name)

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

<?php

namespace MyProject;

 

class MyClass {

    static function static_method()

    {

        echo 'Hello, world!';

    }

}

 

// Unqualified name, resolves to the namespace you are currently in (MyProject\MyClass)

MyClass:static_method();

限定的名字(Qualified Name)

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

<?php

namespace MyProject;

 

require 'myproject/database/connection.php';

 

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

$connection = new Database\Connection();

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

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

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

<?php

namespace MyProject\Database;

 

require 'myproject/fileaccess/input.php';

 

// Trying to access the MyProject\FileAccess\Input class

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

$input = new \MyProject\FileAccess\Input();

对于PHP的内部函数来说,我们不必要使用完全限定的名字。在当前所处的命名空间中,调用一个不存在的未限定的名字的类或函数,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 MyProject\Database 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()!<br />';

}

动态调用
PHP是一门动态语言,也可以将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关键字。

<?php

namespace MyProject;

 

function run() 

{

    echo 'Running from a namespace!';

}

 

// Resolves to MyProject\run

run();

// Explicitly resolves to MyProject\run

namespace\run();

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

<?php

namespace MyProject\Database;

 

// 'MyProject\Database'

echo __NAMESPACE__;

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

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

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

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

<?php

namespace OtherProject;

 

// This holds the MyProject\Database 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 \MyProject\Database\Connection();

 

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

use MyProject\Database\Connection;

 

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

$connection = new Connection();

 

// Import the MyProject\Database namespace

use MyProject\Database;

 

$connection = new Database\Connection()

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

namespace OtherProject;

 

require 'myproject/database/connection.php';

 

use MyProject\Database\Connection as MyConnection;

 

$connection = new MyConnection();

 

use MyProject\Database as MyDatabase;

 

$connection = new MyDatabase\Connection();

总结

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

PHP 相关文章推荐
Windows下的PHP5.0安装配制详解
Sep 05 PHP
dedecms后台验证码总提示错误的解决方法
Mar 21 PHP
dedecms模版制作使用方法
Apr 03 PHP
php中计算未知长度的字符串哪个字符出现的次数最多的代码
Aug 14 PHP
php中的boolean(布尔)类型详解
Oct 28 PHP
简单的php中文转拼音的实现代码
Feb 11 PHP
探寻PHP脚本不报错的原因
Jun 12 PHP
推荐几个开源的微信开发项目
Dec 28 PHP
php中 $$str 中 &quot;$$&quot; 的详解
Jul 06 PHP
PHP页面转UTF-8中文编码乱码的解决办法
Oct 20 PHP
PHP中子类重载父类的方法【parent::方法名】
May 06 PHP
PhpStorm连接服务器并实现自动上传功能
Dec 09 PHP
PHP+JS实现大规模数据提交的方法
Jul 02 #PHP
PHP中iconv函数知识汇总
Jul 02 #PHP
php统计数组元素个数的方法
Jul 02 #PHP
mod_php、FastCGI、PHP-FPM等PHP运行方式对比
Jul 02 #PHP
PHP中的流(streams)浅析
Jul 02 #PHP
PHP curl使用实例
Jul 02 #PHP
PHP中使用curl入门教程
Jul 02 #PHP
You might like
php读取mysql中文数据出现乱码的解决方法
2013/08/16 PHP
织梦sitemap地图实时推送给百度的教程
2015/08/03 PHP
PHP几个实用自定义函数小结
2016/01/25 PHP
PHP编写学校网站上新生注册登陆程序的实例分享
2016/03/21 PHP
PHP Laravel中的Trait使用方法
2019/01/20 PHP
详解PHP PDO简单教程
2019/05/28 PHP
完美解决JS中汉字显示乱码问题(已解决)
2006/12/27 Javascript
在标题栏显示新消息提示,很多公司项目中用到这个方法
2011/11/04 Javascript
JS限制文本框只能输入数字和字母方法
2015/02/28 Javascript
javascript精确统计网站访问量实例代码
2015/12/19 Javascript
Jquery与Bootstrap实现后台管理页面增删改查功能示例
2017/01/22 Javascript
vue添加axios,并且指定baseurl的方法
2018/09/19 Javascript
用Cordova打包Vue项目的方法步骤
2019/02/02 Javascript
Easyui 关闭jquery-easui tab标签页前触发事件的解决方法
2019/04/28 jQuery
基于vue实现一个神奇的动态按钮效果
2019/05/15 Javascript
使用layer弹窗,制作编辑User信息页面的方法
2019/09/27 Javascript
微信小程序swiper左右扩展各显示一半代码实例
2019/12/05 Javascript
Openlayers实现点闪烁扩散效果
2020/09/24 Javascript
vue+Element-ui前端实现分页效果
2020/11/15 Javascript
node.js 基于 STMP 协议和 EWS 协议发送邮件
2021/02/14 Javascript
[01:53]3.19 DOTA2发布会 现场精彩Coser表演
2014/03/25 DOTA
[02:20]2014DOTA2西雅图邀请赛 MVP外卡赛首胜采访
2014/07/09 DOTA
[08:06]DOTA2-DPC中国联赛 正赛 PSG.LGD vs Elephant 选手采访
2021/03/11 DOTA
Python引用传值概念与用法实例小结
2017/10/07 Python
Python3学习urllib的使用方法示例
2017/11/29 Python
python3 map函数和filter函数详解
2019/08/26 Python
Pycharm Plugins加载失败问题解决方案
2020/11/28 Python
HTML5为输入框添加语音输入功能的实现方法
2017/02/06 HTML / CSS
NBA欧洲商店(英国):NBA Europe Store UK
2018/07/27 全球购物
美国最大的电子宠物训练产品制造商:PetSafe
2018/10/12 全球购物
PHP中如何使用Cookie
2015/10/28 面试题
日语专业毕业生求职信
2013/12/04 职场文书
社团活动总结怎么写
2014/06/30 职场文书
学校节能宣传周活动总结
2014/07/09 职场文书
依法行政工作汇报
2014/10/28 职场文书
Python进行区间取值案例讲解
2021/08/02 Python