PHP 5.3新特性命名空间规则解析及高级功能


Posted in PHP onMarch 11, 2010

日前发布的PHP 5.3中,最重要的一个新特性就是命名空间的加入。本文介绍了PHP命名空间的一些术语,其解析规则,以及一些高级功能的应用,希望能够帮助读者在项目中真正使用命名空间。

在这里中我们介绍了PHP命名空间的用途和namespace关键字,在这篇文章中我们将介绍一下use命令的使用以及PHP如何解析命名空间的名字的。

为了便于对比,我定义了两个几乎一样的代码块,只有命名空间的名字不同。

lib2.php

 

开始之前先要理解几个PHP命名空间相关术语。

◆完全限定名称(Fully-qualified name)

任何PHP代码都可以引用完全限定名称,它是一个以命名空间反斜线开头的标识符,如\App\Lib1\MYCONST,\App\Lib2\MyFunction( )等。

完全限定名称是没有任何歧义的,开头的反斜线和文件路径的作用有点类似,它表示“根”全局空间,如果我们在全局空间中实现了一个不同的MyFunction( ),可以使用\MyFunction( )从lib1.php或lib2.php调用它。

完全限定名称对一次性函数调用或对象初始化非常有用,但当你产生了大量的调用时它们就没有实用价值了,在下面的讨论中我们将会看到,PHP提供了其它选项以解除我们为命名空间打字的烦恼。

◆限定名称(Qualified name)

至少有一个命名空间分隔符的标识符,如Lib1\MyFunction( )。

◆非限定名称(Unqualified name)

没有命名空间分隔符的标识符,如MyFunction( )。

在相同的命名空间内工作

仔细思考下面的代码:

myapp1.php

 

即使我们同时包括了lib1.php和lib2.php,MYCONST,MyFunction和MyClass标识符只能在lib1.php中引用,这是因为myapp1.php的代码在相同的App\Lib1命名空间内。

执行结果:

  1. App\Lib1\MYCONST  
  2. App\Lib1\MyFunction  
  3. App\Lib1\MyClass::WhoAmI  

命名空间导入

可以使用use操作符导入命名空间,如:

myapp2.php

 
 

可以定义任意数量的use语句,或使用逗号分隔成独立的命名空间,在这个例子中我们导入了App\Lib2命名空间,但我们仍然不能直接引用 MYCONST,MyFunction和MyClass,因为我们的代码还在全局空间中,但如果我们添加了“Lib2\”前缀,它们就变成限定名称 了,PHP将会搜索导入的命名空间,直到找到匹配项。

执行结果:

  1. App\Lib2\MYCONST  
  2. App\Lib2\MyFunction  
  3. App\Lib2\MyClass::WhoAmI 

命名空间别名

命名空间别名可能是最有用的构想了,别名允许我们使用较短的名称引用很长的命名空间。

myapp3.php

 

第一个use语句将App\Lib1定义为“L”,任何使用“L”的限定名称在编译时都会被翻译成“App\Lib1”,因此我们就可以引用L\MYCONST和L\MyFunction而不是完全限定名称了。

第二个use语句定义了“obj”作为App\Lib2\命名空间中MyClass类的别名,这种方式只适合于类,不能用于常量和函数,现在我们就可以使用new Obj( )或象上面那样运行静态方法了。

执行结果:

  1. App\Lib1\MYCONST  
  2. App\Lib1\MyFunction  
  3. App\Lib1\MyClass::WhoAmI  
  4. App\Lib2\MyClass::WhoAmI  

PHP命名解析规则

PHP标识符名称使用下列命名空间规则进行解析,请参考PHP用户手册了解更详细的信息:

1.在编译时调用完全限定函数、类或常量;

2.非限定名称和限定名称根据导入规则进行翻译,例如,如果A\B\C导入为C,调用C\D\e( )就会被翻译成A\B\C\D\e( );

3.在PHP命名空间内,所有限定名称尚未根据导入规则转换,例如,如果在命名空间A\B中调用C\D\e( ),那么会被翻译成A\B\C\D\e( );

4.非限定类名称根据当前的导入规则进行转换,使用全名替换导入的短名称,例如,如果类C在命名空间A\B中被导入为X,那么new X( )就会被翻译为new A\B\C( );

5.在命名空间中非限定函数调用在运行时解析,例如,如果MyFunction( )在命名空间A\B中被调用,PHP首先会查找函数\A\B\MyFunction( ),如果没有找到,然后会在全局空间中查找\MyFunction( );

6.调用非限定或限定类名在运行时被解析,例如,如果我们在命名空间A\B中调用new C( ),PHP将会查找类A\B\C,如果没有找到,PHP会尝试自动载入A\B\C。

PHP命名空间高级特性

接下来让我们看一看PHP命名空间的一些高级特性。

__NAMESPACE__常量

__NAMESPACE__是一个PHP字符串,它总是返回当前命名空间的名称,在全局空间中它是一个空字符串。

   

这个值在调试时非常有用,它也可由于动态生成一个完全限定类名,如:

WhoAmI(); // outputs: App\Lib1\MyClass::WhoAmI   ?>  

namespace关键字

namespace关键字可以用于明确引用一个当前命名空间或子命名空间中的项目,它等价于类中的self命名空间:

WhoAmI(); // outputs: App\Lib1\MyClass::WhoAmI   ?>  

自动载入命名空间类

PHP 5中最省时省力的特性是自动载入,在全局(非命名空间)PHP代码中,可以写一个标准自动载入函数:

 

在PHP 5.3中,你可以创建一个命名空间类的实例,在这种情况下,完全限定命名空间和类名传递给__autoload函数,例如,$class_name的值可 能是App\Lib1\MyClass。你可以在相同的文件夹下放置所有的PHP类文件,从字符串中提取命名空间,但那样会导致文件名冲突。

另外,你的类文件层次结构会按照命名空间的结构重新组织,例如,MyClass.php文件可以创建在/classes/App/Lib1文件夹下:

/classes/App/Lib1/MyClass.php

 

在根文件夹下的文件就使用下面的代码了:

myapp.php

WhoAmI();     // autoload function   function __autoload($class) {    // convert namespace to full file path    $class = 'classes/' . str_replace('\\', '/', $class) . '.php';    require_once($class);   }   ?>  

解释:

1.类App\Lib1\MyClass的别名是MC;

2. new MC( )在编译时被翻译成new App\Lib1\MyClass( );

3.字符串App\Lib1\MyClass被传递给__autoload函数,使用文件路径正斜线替换所有命名空间中的反斜线,然后修改字符串,classes\App\Lib1\MyClass.php文件被自动载入;

总结

有关PHP命名空间的使用就介绍到这里,希望您能够对PHP的命名空间有一个新的认识,并希望你能在新项目中真正使用命名空间。

PHP 相关文章推荐
mysql5详细安装教程
Jan 15 PHP
用PHP连接MySQL代码的参数说明
Jun 07 PHP
PHP 批量更新网页内容实现代码
Jan 05 PHP
Destoon模板制作简明教程
Jun 20 PHP
PHP实现多图片上传类实例
Jul 26 PHP
PHP+FastCGI+Nginx配置PHP运行环境
Aug 07 PHP
Laravel框架表单验证详解
Sep 04 PHP
PHP正则替换函数preg_replace和preg_replace_callback使用总结
Sep 22 PHP
使用Thinkphp框架开发移动端接口
Aug 05 PHP
php 参数过滤、数据过滤详解
Oct 26 PHP
php+MySql实现登录系统与输出浏览者信息功能
Jul 01 PHP
Mac下快速搭建PHP开发环境步骤详解
May 05 PHP
PHP Memcached + APC + 文件缓存封装实现代码
Mar 11 #PHP
了解Joomla 这款来自国外的php网站管理系统
Mar 11 #PHP
PHP调用Twitter的RSS的实现代码
Mar 10 #PHP
PHP中include()与require()的区别说明
Mar 10 #PHP
PHP扩展编写点滴 技巧收集
Mar 09 #PHP
php 修改zen-cart下单和付款流程以防止漏单
Mar 08 #PHP
PHP 最大运行时间 max_execution_time修改方法
Mar 08 #PHP
You might like
php读取txt文件组成SQL并插入数据库的代码(原创自Zjmainstay)
2012/07/31 PHP
PHP Global变量定义当前页面的全局变量实现探讨
2013/06/05 PHP
ThinkPHP验证码和分页实例教程
2014/08/22 PHP
PHP读取大文件的几种方法介绍
2016/10/27 PHP
PHP实现与java 通信的插件使用教程
2019/08/11 PHP
php DES加密算法实例分析
2019/09/18 PHP
javascript 写类方式之一
2009/07/05 Javascript
JQuery扩展插件Validate—4设置错误提示的样式
2011/09/05 Javascript
表头固定(利用jquery实现原理介绍)
2012/11/08 Javascript
jquery如何通过name名称获取当前name的value值
2013/12/20 Javascript
jQuery事件用法实例汇总
2014/08/29 Javascript
AngularJS入门教程一:路由用法初探
2017/05/27 Javascript
javascript回调函数的概念理解与用法分析
2017/05/27 Javascript
基于JavaScript实现多级菜单效果
2017/07/25 Javascript
vue-router2.0 组件之间传参及获取动态参数的方法
2017/11/10 Javascript
jQuery实现表单动态加减、ajax表单提交功能
2018/06/08 jQuery
javascrit中undefined和null的区别详解
2019/04/07 Javascript
Vue.js页面中有多个input搜索框如何实现防抖操作
2019/11/04 Javascript
微信小程序获取复选框全选反选选中的值(实例代码)
2019/12/17 Javascript
[59:08]DOTA2上海特级锦标赛C组小组赛#2 LGD VS Newbee第一局
2016/02/27 DOTA
[01:03:47]VP vs NewBee Supermajor 胜者组 BO3 第一场 6.5
2018/06/06 DOTA
[59:48]LGD vs IG 2018国际邀请赛小组赛BO2 第一场 8.18
2018/08/19 DOTA
Python排序搜索基本算法之选择排序实例分析
2017/12/09 Python
python 动态加载的实现方法
2017/12/22 Python
Flask教程之重定向与错误处理实例分析
2019/08/01 Python
美国知名保健品网站:LuckyVitamin(支持中文)
2017/08/09 全球购物
校园奶茶店创业计划书
2014/01/23 职场文书
致跳远运动员加油稿
2014/02/11 职场文书
巾帼文明岗申报材料
2014/05/01 职场文书
追讨欠款律师函
2015/06/24 职场文书
导游词之山东八仙过海景区
2019/11/11 职场文书
python使用pygame创建精灵Sprite
2021/04/06 Python
为什么node.js不适合大型项目
2021/04/28 Javascript
mysql 索引合并的使用
2021/08/30 MySQL
5种 JavaScript 方式实现数组扁平化
2021/10/05 Javascript
基于HTML十秒做出淘宝页面
2021/10/24 HTML / CSS