找回密码
 立即注册

QQ登录

只需一步,快速开始

查看: 1514|回复: 0
打印 上一主题 下一主题

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

[复制链接]

2536

主题

2536

帖子

7532

积分

论坛元老

Rank: 8Rank: 8

积分
7532
跳转到指定楼层
楼主
发表于 2018-2-14 09:00:57 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

            日前发布的PHP 5.3中,最重要的一个新特性就是命名空间的加入。本文介绍了PHP命名空间的一些术语,其解析规则,以及一些高级功能的应用,希望能够帮助读者在项目中真正使用命名空间。
在这里中我们介绍了PHP命名空间的用途和namespace关键字,在这篇文章中我们将介绍一下use命令的使用以及PHP如何解析命名空间的名字的。
为了便于对比,我定义了两个几乎一样的代码块,只有命名空间的名字不同。
[TR]
[TD]
[ol]


  • // application library 1
      
    namespace App\Lib1;   

    const
    MYCONST =
    [color=]'App\Lib1\MYCONST'
    ;   

    function
    MyFunction() {   

  • return

    __FUNCTION__
    ;   
    }   

    class
    MyClass {   

  • static

    function
    WhoAmI() {   
    eturn
    __METHOD__
    ;   
  • }   
  • }   
  • ?> [/ol][/TD][/TR]

  • lib2.php
    [TR]
    [TD]
    [ol]


  • // application library 2
      
    namespace App\Lib2;   


  • const
    MYCONST =
    [color=]'App\Lib2\MYCONST'
    ;   


  • function
    MyFunction() {   

  • return

    __FUNCTION__
    ;   
    }   


  • class
    MyClass {   

  • static

    function
    WhoAmI() {   
    eturn
    __METHOD__
    ;   
  • }   
  • }   
  • ?>  [/ol][/TD][/TR]

  • 开始之前先要理解几个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
    [TR]
    [TD]
  • namespace App\Lib1;   

  • require_once
    (
    [color=]'lib1.php'
    );  
    require_once
    (
    [color=]'lib2.php'
    );  
  • header(
  • [color=]'Content-type: text/plain'
    );  
  • echo MYCONST .
  • [color=]"
    "
    ;  echo MyFunction() .
    [color=]"
    "
    ;  
  • echo MyClass::WhoAmI() .
  • [color=]"
    "
    ;  ?>  
    [/ol][/TD][/TR]

    即使我们同时包括了lib1.php和lib2.php,MYCONST,MyFunction和MyClass标识符只能在lib1.php中引用,这是因为myapp1.php的代码在相同的App\Lib1命名空间内。
    执行结果:
    App\Lib1\MYCONST   
  • App\Lib1\MyFunction   
    App\Lib1\MyClass::WhoAmI   [/ol]
    命名空间导入
    可以使用use操作符导入命名空间,如:
    myapp2.php
    [TR]
    [TD]  
    [ol]

  • use
    App\Lib2;   

  • require_once
    (
    [color=]'lib1.php'
    );   

    require_once
    (
    [color=]'lib2.php'
    );   

  • header(
    [color=]'Content-type: text/plain'
    );   
  • echo Lib2\MYCONST .
  • [color=]"
    "
    ;   
    echo Lib2\MyFunction() .
    [color=]"
    "
    ;   
  • echo Lib2\MyClass::WhoAmI() .
  • [color=]"
    "
    ;   
    ?> [/ol][/TD][/TR]

    可以定义任意数量的use语句,或使用逗号分隔成独立的命名空间,在这个例子中我们导入了App\Lib2命名空间,但我们仍然不能直接引用 MYCONST,MyFunction和MyClass,因为我们的代码还在全局空间中,但如果我们添加了“Lib2\”前缀,它们就变成限定名称 了,PHP将会搜索导入的命名空间,直到找到匹配项。
    执行结果:
    App\Lib2\MYCONST   
  • App\Lib2\MyFunction   
    App\Lib2\MyClass::WhoAmI  [/ol]
    命名空间别名
    命名空间别名可能是最有用的构想了,别名允许我们使用较短的名称引用很长的命名空间。
    myapp3.php
    [TR]
    [TD]


  • use
    App\Lib1
    as
    L;   

    use
    App\Lib2\MyClass
    as
    Obj;   

  • header(
    [color=]'Content-type: text/plain'
    );   

  • require_once
    (
    [color=]'lib1.php'
    );   

    require_once
    (
    [color=]'lib2.php'
    );   

  • echo L\MYCONST .
    [color=]"
    "
    ;   
  • echo L\MyFunction() .
  • [color=]"
    "
    ;   
    echo L\MyClass::WhoAmI() .
    [color=]"
    "
    ;   
  • echo Obj::WhoAmI() .
  • [color=]"
    "
    ;   
    ?>   [/ol][/TD][/TR]

    第一个use语句将App\Lib1定义为“L”,任何使用“L”的限定名称在编译时都会被翻译成“App\Lib1”,因此我们就可以引用L\MYCONST和L\MyFunction而不是完全限定名称了。
    第二个use语句定义了“obj”作为App\Lib2\命名空间中MyClass类的别名,这种方式只适合于类,不能用于常量和函数,现在我们就可以使用new Obj( )或象上面那样运行静态方法了。
    执行结果:
    App\Lib1\MYCONST   
  • App\Lib1\MyFunction   
    App\Lib1\MyClass::WhoAmI   
  • App\Lib2\MyClass::WhoAmI   [/ol]
    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字符串,它总是返回当前命名空间的名称,在全局空间中它是一个空字符串。
    [TR]
    [TD]
  • namespace App\Lib1;  echo __NAMESPACE__;
  • // outputs: App\Lib1
  • ?>   [/ol][/TD][/TR]

  • 这个值在调试时非常有用,它也可由于动态生成一个完全限定类名,如:
    [TR]
    [TD]
  • namespace App\Lib1;   

  • class
    MyClass {   
    public

    function
    WhoAmI() {  

  • return

    __METHOD__
    ;   }  
  • }   

  • [color=]$c
    = __NAMESPACE__ .
    [color=]'\\MyClass'
    ;  
    [color=]$m
    =
    new

    [color=]$c
    ;  
  • echo
  • [color=]$m
    ->WhoAmI();
    // outputs: App\Lib1\MyClass::WhoAmI
    ?>  [/ol][/TD][/TR]

    namespace关键字
    namespace关键字可以用于明确引用一个当前命名空间或子命名空间中的项目,它等价于类中的self命名空间:
    [TR]
    [TD]
  • namespace App\Lib1;   

  • class
    MyClass {   
    public

    function
    WhoAmI() {  

  • return

    __METHOD__
    ;   }  
  • }   

  • [color=]$m
    =
    new
    namespace\MyClass;  echo
    [color=]$m
    ->WhoAmI();
    // outputs: App\Lib1\MyClass::WhoAmI
  • ?>  [/ol][/TD][/TR]

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

  • [color=]$obj
    =
    new
    MyClass1();
    // classes/MyClass1.php is auto-loaded

    [color=]$obj
    =
    new
    MyClass2();
    // classes/MyClass2.php is auto-loaded

  • // autoload function

  • function
    __autoload(
    [color=]$class_name
    ) {   
    require_once
    (
    [color=]"classes/$class_name.php"
    );  
  • }  ?>  [/ol][/TD][/TR]

  • 在PHP 5.3中,你可以创建一个命名空间类的实例,在这种情况下,完全限定命名空间和类名传递给__autoload函数,例如,$class_name的值可 能是App\Lib1\MyClass。你可以在相同的文件夹下放置所有的PHP类文件,从字符串中提取命名空间,但那样会导致文件名冲突。
    另外,你的类文件层次结构会按照命名空间的结构重新组织,例如,MyClass.php文件可以创建在/classes/App/Lib1文件夹下:
    /classes/App/Lib1/MyClass.php
    [TR]
    [TD]
  • namespace App\Lib1;   

  • class
    MyClass {   
    public

    function
    WhoAmI() {  

  • return

    __METHOD__
    ;   }  
  • }  ?>  [/ol][/TD][/TR]

  • 在根文件夹下的文件就使用下面的代码了:
    myapp.php
    [TR]
    [TD]

  • use
    App\Lib1\MyClass
    as
    MC;   

  • [color=]$obj
    =
    new
    MC();  echo
    [color=]$obj
    ->WhoAmI();  

  • // autoload function

  • function
    __autoload(
    [color=]$class
    ) {   
    // convert namespace to full file path

  • [color=]$class
    =
    [color=]'classes/'
    . str_replace(
    [color=]'\\', '
    /
    [color=]', $class) . '
    .php';   
    require_once
    (
    [color=]$class
    );  
  • }  ?>  [/ol][/TD][/TR]

  • 解释:
    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命名空间(Namespace)的使用详解
  • php命名空间学习详解
  • PHP命名空间(Namespace)简明教程
  • PHP命名空间(namespace)的使用基础及示例
  • PHP命名空间(namespace)的动态访问及使用技巧
  • PHP中的命名空间详细介绍
  • thinkphp autoload 命名空间自定义 namespace
  • php中namespace use用法实例分析
  • PHP命名空间和自动加载类
  • PHP命名空间namespace用法实例分析
  • php中namespace及use用法分析
  • PHP命名空间namespace的定义方法详解
            
  • 分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    用户反馈
    客户端