当前位置:首页 > PHP教程 > php高级应用 > 列表

PHP命名空间与自动加载机制的基础介绍

发布:smiling 来源: PHP粉丝网  添加日期:2021-12-12 15:21:30 浏览: 评论:0 

这篇文章主要给大家介绍了关于PHP命名空间与自动加载机制的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用PHP具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧。

前言

include 和 require 是PHP中引入文件的两个基本方法。在小规模开发中直接使用 include 和 require 没哟什么不妥,但在大型项目中会造成大量的 include 和 require 堆积。这样的代码既不优雅,执行效率也很低,而且维护起来也相当困难。

为了解决这个问题,部分框架会给出一个引入文件的配置清单,在对象初始化的时候把需要的文件引入。但这只是让代码变得更简洁了一些,引入的效果仍然是差强人意。PHP5 之后,随着 PHP 面向对象支持的完善,__autoload 函数才真正使得自动加载成为可能。

* include 和 require 功能是一样的,它们的不同在于 include 出错时只会产生警告,而 require 会抛出错误终止脚本。

* include_once 和 include 唯一的区别在于 include_once 会检查文件是否已经引入,如果是则不会重复引入。

=================自动加载==================

实现自动加载最简单的方式就是使用 __autoload 魔术方法。当需要使用的类没有被引入时,这个函数会在PHP报错前被触发,未定义的类名会被当作参数传入。至于函数具体的逻辑,这需要用户自己去实现。

首先创建一个 autoload.php 来做一个简单的测试:

  1. // 类未定义时,系统自动调用 
  2. function __autoload($class
  3.  /* 具体处理逻辑 */ 
  4.  echo $class;// 简单的输出未定义的类名 
  5.  
  6. new HelloWorld(); 
  7.  
  8. /** 
  9.  * 输出 HelloWorld 与报错信息 
  10.  * Fatal error: Class 'HelloWorld' not found 
  11.  */ 

通过这个简单的例子可以发现,在类的实例化过程中,系统所做的工作大致是这样的:

  1. /* 模拟系统实例化过程 */ 
  2. function instance($class
  3.  // 如果类存在则返回其实例 
  4.  if (class_exists($class, false)) { 
  5.   return new $class(); 
  6.  } 
  7.  // 查看 autoload 函数是否被用户定义 
  8.  if (function_exists('__autoload')) { 
  9.   __autoload($class); // 最后一次引入的机会 
  10.  } 
  11.  // 再次检查类是否存在 
  12.  if (class_exists($class, false)) { 
  13.   return new $class(); 
  14.  } else { // 系统:我实在没辙了 
  15.   throw new Exception('Class Not Found'); 
  16.  } 

明白了 __autoload 函数的工作原理之后,那就让我们来用它去实现自动加载。

首先创建一个类文件(建议文件名与类名一致),代码如下:

  1. class [ClassName]  
  2.  // 对象实例化时输出当前类名 
  3.  function __construct() 
  4.  { 
  5.   echo '<h1>' . __CLASS__ . '</h1>'
  6.  } 

(我这里创建了一个 HelloWorld 类用作演示)接下来我们就要定义 __autoload 的具体逻辑,使它能够实现自动加载:

  1. function __autoload($class
  2.  // 根据类名确定文件名 
  3.  $file = $class . '.php'
  4.  
  5.  if (file_exists($file)) { 
  6.   include $file// 引入PHP文件 
  7.  } 
  8.  
  9. new HelloWorld(); 
  10.  
  11. /** 
  12.  * 输出 <h1>HelloWorld</h1> 
  13.  */ 

=================命名空间==================

其实命名空间并不是什么新生事物,很多语言(例如C++)早都支持这个特性了。只不过 PHP 起步比较晚,直到 PHP 5.3 之后才支持。

命名空间简而言之就是一种标识,它的主要目的是解决命名冲突的问题。

就像在日常生活中,有很多姓名相同的人,如何区分这些人呢?那就需要加上一些额外的标识。

把工作单位当成标识似乎不错,这样就不用担心 “撞名” 的尴尬了。

这里我们来做一个小任务,去介绍百度的CEO李彦宏:

  1. namespace 百度; 
  2.  
  3. class 李彦宏 
  4.  function __construct() 
  5.  { 
  6.   echo '百度创始人'
  7.  } 

↑ 这就是李彦宏的基本资料了,namespace 是他的单位标识,class 是他的姓名。

命名空间通过关键字 namespace 来声明。如果一个文件中包含命名空间,它必须在其它所有代码之前声明命名空间。

new 百度\李彦宏(); // 限定类名

new \百度\李彦宏(); // 完全限定类名

↑ 在一般情况下,无论是向别人介绍 "百度 李彦宏" 还是 "百度公司 李彦宏",他们都能够明白。

在当前命名空间没有声明的情况下,限定类名和完全限定类名是等价的。因为如果不指定空间,则默认为全局(\)。

namespace 谷歌;

new 百度\李彦宏(); // 谷歌\百度\李彦宏(实际结果)

new \百度\李彦宏(); // 百度\李彦宏(实际结果)

↑ 如果你在谷歌公司向他们的员工介绍李彦宏,一定要指明是 "百度公司的李彦宏"。否则他会认为百度是谷歌的一个部门,而李彦宏只是其中的一位员工而已。

这个例子展示了在命名空间下,使用限定类名和完全限定类名的区别。(完全限定类名 = 当前命名空间 + 限定类名)

  1. /* 导入命名空间 */ 
  2. use 百度\李彦宏; 
  3. new 李彦宏(); // 百度\李彦宏(实际结果) 
  4.  
  5. /* 设置别名 */ 
  6. use 百度\李彦宏 AS CEO; 
  7. new CEO(); // 百度\李彦宏(实际结果) 
  8.  
  9. /* 任何情况 */ 
  10. new \百度\李彦宏();// 百度\李彦宏(实际结果) 

↑ 第一种情况是别人已经认识李彦宏了,你只需要直接说名字,他就能知道你指的是谁。第二种情况是李彦宏就是他们的CEO,你直接说CEO,他可以立刻反应过来。

使用命名空间只是让类名有了前缀,不容易发生冲突,系统仍然不会进行自动导入。

如果不引入文件,系统会在抛出 "Class Not Found" 错误之前触发 __autoload 函数,并将限定类名传入作为参数。

所以上面的例子都是基于你已经将相关文件手动引入的情况下实现的,否则系统会抛出 " Class '百度\李彦宏' not found"。

=================spl_autoload==================

接下来让我们要在含有命名空间的情况下去实现自动加载。这里我们使用 spl_autoload_register() 函数来实现,这需要你的 PHP 版本号大于 5.12。

spl_autoload_register 函数的功能就是把传入的函数(参数可以为回调函数或函数名称形式)注册到 SPL __autoload 函数队列中,并移除系统默认的 __autoload() 函数。

一旦调用 spl_autoload_register() 函数,当调用未定义类时,系统就会按顺序调用注册到 spl_autoload_register() 函数的所有函数,而不是自动调用 __autoload() 函数。

现在,我们来创建一个 Linux 类,它使用 os 作为它的命名空间(建议文件名与类名保持一致):

  1. namespace os; // 命名空间 
  2.  
  3. class Linux // 类名 
  4.  function __construct() 
  5.  { 
  6.   echo '<h1>' . __CLASS__ . '</h1>'
  7.  } 

接着,在同一个目录下新建一个 PHP 文件,使用 spl_autoload_register 以函数回调的方式实现自动加载:

  1. spl_autoload_register(function ($class) { // class = os\Linux 
  2.  
  3.  /* 限定类名路径映射 */ 
  4.  $class_map = array
  5.   // 限定类名 => 文件路径 
  6.   'os\\Linux' => './Linux.php'
  7.  ); 
  8.  
  9.  /* 根据类名确定文件名 */ 
  10.  $file = $class_map[$class]; 
  11.  
  12.  /* 引入相关文件 */ 
  13.  if (file_exists($file)) { 
  14.   include $file
  15.  } 
  16. }); 
  17.  
  18. new \os\Linux(); 

这里我们使用了一个数组去保存类名与文件路径的关系,这样当类名传入时,自动加载器就知道该引入哪个文件去加载这个类了。

但是一旦文件多起来的话,映射数组会变得很长,这样的话维护起来会相当麻烦。如果命名能遵守统一的约定,就可以让自动加载器自动解析判断类文件所在的路径。接下来要介绍的PSR-4 就是一种被广泛采用的约定方式。

=================PSR-4规范==================

PSR-4 是关于由文件路径自动载入对应类的相关规范,规范规定了一个完全限定类名需要具有以下结构:

\<顶级命名空间>(\<子命名空间>)*\<类名>

如果继续拿上面的例子打比方的话,顶级命名空间相当于公司,子命名空间相当于职位,类名相当于人名。那么李彦宏标准的称呼为 "百度公司 CEO 李彦宏"。

PSR-4 规范中必须要有一个顶级命名空间,它的意义在于表示某一个特殊的目录(文件基目录)。子命名空间代表的是类文件相对于文件基目录的这一段路径(相对路径),类名则与文件名保持一致(注意大小写的区别)。

举个例子:在全限定类名 \app\view\news\Index 中,如果 app 代表 C:\Baidu,那么这个类的路径则是 C:\Baidu\view\news\Index.php

我们就以解析 \app\view\news\Index 为例,编写一个简单的 Demo:

  1. $class = 'app\view\news\Index'
  2.  
  3. /* 顶级命名空间路径映射 */ 
  4. $vendor_map = array
  5.   'app' => 'C:\Baidu'
  6. ); 
  7.  
  8. /* 解析类名为文件路径 */ 
  9. $vendor = substr($class, 0, strpos($class'\\')); // 取出顶级命名空间[app] 
  10. $vendor_dir = $vendor_map[$vendor]; // 文件基目录[C:\Baidu] 
  11. $rel_path = dirname(substr($classstrlen($vendor))); // 相对路径[/view/news] 
  12. $file_name = basename($class) . '.php'// 文件名[Index.php] 
  13.  
  14. /* 输出文件所在路径 */ 
  15. echo $vendor_dir . $rel_path . DIRECTORY_SEPARATOR . $file_name

通过这个 Demo 可以看出限定类名转换为路径的过程。那么现在就让我们用规范的面向对象方式去实现自动加载器吧。

首先我们创建一个文件 Index.php,它处于 \app\mvc\view\home 目录中:

  1. namespace app\mvc\view\home; 
  2.  
  3. class Index 
  4.   function __construct() 
  5.   { 
  6.     echo '<h1> Welcome To Home </h1>'
  7.   } 

接着我们在创建一个加载类(不需要命名空间),它处于 \ 目录中:

  1. class Loader 
  2.   /* 路径映射 */ 
  3.   public static $vendorMap = array
  4.     'app' => __DIR__ . DIRECTORY_SEPARATOR . 'app'
  5.   ); 
  6.  
  7.   /** 
  8.    * 自动加载器 
  9.    */ 
  10.   public static function autoload($class
  11.   { 
  12.     $file = self::findFile($class); 
  13.     if (file_exists($file)) { 
  14.       self::includeFile($file); 
  15.     } 
  16.   } 
  17.  
  18.   /** 
  19.    * 解析文件路径 
  20.    */ 
  21.   private static function findFile($class
  22.   { 
  23.     $vendor = substr($class, 0, strpos($class'\\')); // 顶级命名空间 
  24.     $vendorDir = self::$vendorMap[$vendor]; // 文件基目录 
  25.     $filePath = substr($classstrlen($vendor)) . '.php'// 文件相对路径 
  26.     return strtr($vendorDir . $filePath'\\', DIRECTORY_SEPARATOR); // 文件标准路径 
  27.   } 
  28.  
  29.   /** 
  30.    * 引入文件 
  31.    */ 
  32.   private static function includeFile($file
  33.   { 
  34.     if (is_file($file)) { 
  35.       include $file
  36.     } 
  37.   } 

最后,将 Loader 类中的 autoload 注册到 spl_autoload_register 函数中:

  1. include 'Loader.php'// 引入加载器 
  2. spl_autoload_register('Loader::autoload'); // 注册自动加载 
  3.  
  4. new \app\mvc\view\home\Index(); // 实例化未引用的类 
  5.  
  6. /** 
  7.  * 输出: <h1> Welcome To Home </h1> 
  8.  */ 

示例中的代码其实就是 ThinkPHP 自动加载器源码的精简版,它是 ThinkPHP 5 能实现惰性加载的关键。

至此,自动加载的原理已经全部讲完了,如果有兴趣深入了解的话,可以参考下面的 ThinkPHP 源码。

  1. class Loader 
  2.   protected static $instance = []; 
  3.   // 类名映射 
  4.   protected static $map = []; 
  5.  
  6.   // 命名空间别名 
  7.   protected static $namespaceAlias = []; 
  8.  
  9.   // PSR-4 
  10.   private static $prefixLengthsPsr4 = []; 
  11.   private static $prefixDirsPsr4  = []; 
  12.   private static $fallbackDirsPsr4 = []; 
  13.  
  14.   // PSR-0 
  15.   private static $prefixesPsr0   = []; 
  16.   private static $fallbackDirsPsr0 = []; 
  17.  
  18.   // 自动加载的文件 
  19.   private static $autoloadFiles = []; 
  20.  
  21.   // 自动加载 
  22.   public static function autoload($class
  23.   { 
  24.     // 检测命名空间别名 
  25.     if (!emptyempty(self::$namespaceAlias)) { 
  26.       $namespace = dirname($class); 
  27.       if (isset(self::$namespaceAlias[$namespace])) { 
  28.         $original = self::$namespaceAlias[$namespace] . '\\' . basename($class); 
  29.         if (class_exists($original)) { 
  30.           return class_alias($original$class, false); 
  31.         } 
  32.       } 
  33.     } 
  34.  
  35.     if ($file = self::findFile($class)) { 
  36.  
  37.       // Win环境严格区分大小写 
  38.       if (IS_WIN && pathinfo($file, PATHINFO_FILENAME) != pathinfo(realpath($file), PATHINFO_FILENAME)) { 
  39.         return false; 
  40.       } 
  41.  
  42.       __include_file($file); 
  43.       return true; 
  44.     } 
  45.   } 
  46.  
  47.   /** 
  48.    * 查找文件 
  49.    * @param $class 
  50.    * @return bool 
  51.    */ 
  52.   private static function findFile($class
  53.   { 
  54.     if (!emptyempty(self::$map[$class])) { 
  55.       // 类库映射 
  56.       return self::$map[$class]; 
  57.     } 
  58.  
  59.     // 查找 PSR-4 
  60.     $logicalPathPsr4 = strtr($class'\\', DS) . EXT; 
  61.  
  62.     $first = $class[0]; 
  63.     if (isset(self::$prefixLengthsPsr4[$first])) { 
  64.       foreach (self::$prefixLengthsPsr4[$firstas $prefix => $length) { 
  65.         if (0 === strpos($class$prefix)) { 
  66.           foreach (self::$prefixDirsPsr4[$prefixas $dir) { 
  67.             if (is_file($file = $dir . DS . substr($logicalPathPsr4$length))) { 
  68.               return $file
  69.             } 
  70.           } 
  71.         } 
  72.       } 
  73.     } 
  74.  
  75.     // 查找 PSR-4 fallback dirs 
  76.     foreach (self::$fallbackDirsPsr4 as $dir) { 
  77.       if (is_file($file = $dir . DS . $logicalPathPsr4)) { 
  78.         return $file
  79.       } 
  80.     } 
  81.  
  82.     // 查找 PSR-0 
  83.     if (false !== $pos = strrpos($class'\\')) { 
  84.       // namespaced class name 
  85.       $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1) 
  86.       . strtr(substr($logicalPathPsr4$pos + 1), '_', DS); 
  87.     } else { 
  88.       // PEAR-like class name 
  89.       $logicalPathPsr0 = strtr($class'_', DS) . EXT; 
  90.     } 
  91.  
  92.     if (isset(self::$prefixesPsr0[$first])) { 
  93.       foreach (self::$prefixesPsr0[$firstas $prefix => $dirs) { 
  94.         if (0 === strpos($class$prefix)) { 
  95.           foreach ($dirs as $dir) { 
  96.             if (is_file($file = $dir . DS . $logicalPathPsr0)) { 
  97.               return $file
  98.             } 
  99.           } 
  100.         } 
  101.       } 
  102.     } 
  103.  
  104.     // 查找 PSR-0 fallback dirs 
  105.     foreach (self::$fallbackDirsPsr0 as $dir) { 
  106.       if (is_file($file = $dir . DS . $logicalPathPsr0)) { 
  107.         return $file
  108.       } 
  109.     } 
  110.  
  111.     return self::$map[$class] = false; 
  112.   } 
  113.  
  114.   // 注册classmap 
  115.   public static function addClassMap($class$map = ''
  116.   { 
  117.     if (is_array($class)) { 
  118.       self::$map = array_merge(self::$map$class); 
  119.     } else { 
  120.       self::$map[$class] = $map
  121.     } 
  122.   } 
  123.  
  124.   // 注册命名空间 
  125.   public static function addNamespace($namespace$path = ''
  126.   { 
  127.     if (is_array($namespace)) { 
  128.       foreach ($namespace as $prefix => $paths) { 
  129.         self::addPsr4($prefix . '\\', rtrim($paths, DS), true); 
  130.       } 
  131.     } else { 
  132.       self::addPsr4($namespace . '\\', rtrim($path, DS), true); 
  133.     } 
  134.   } 
  135.  
  136.   // 添加Ps0空间 
  137.   private static function addPsr0($prefix$paths$prepend = false) 
  138.   { 
  139.     if (!$prefix) { 
  140.       if ($prepend) { 
  141.         self::$fallbackDirsPsr0 = array_merge
  142.           (array$paths
  143.           self::$fallbackDirsPsr0 
  144.         ); 
  145.       } else { 
  146.         self::$fallbackDirsPsr0 = array_merge
  147.           self::$fallbackDirsPsr0
  148.           (array$paths 
  149.         ); 
  150.       } 
  151.  
  152.       return
  153.     } 
  154.  
  155.     $first = $prefix[0]; 
  156.     if (!isset(self::$prefixesPsr0[$first][$prefix])) { 
  157.       self::$prefixesPsr0[$first][$prefix] = (array$paths
  158.  
  159.       return
  160.     } 
  161.     if ($prepend) { 
  162.       self::$prefixesPsr0[$first][$prefix] = array_merge
  163.         (array$paths
  164.         self::$prefixesPsr0[$first][$prefix
  165.       ); 
  166.     } else { 
  167.       self::$prefixesPsr0[$first][$prefix] = array_merge
  168.         self::$prefixesPsr0[$first][$prefix], 
  169.         (array$paths 
  170.       ); 
  171.     } 
  172.   } 
  173.  
  174.   // 添加Psr4空间 
  175.   private static function addPsr4($prefix$paths$prepend = false) 
  176.   { 
  177.     if (!$prefix) { 
  178.       // Register directories for the root namespace. 
  179.       if ($prepend) { 
  180.         self::$fallbackDirsPsr4 = array_merge
  181.           (array$paths
  182.           self::$fallbackDirsPsr4 
  183.         ); 
  184.       } else { 
  185.         self::$fallbackDirsPsr4 = array_merge
  186.           self::$fallbackDirsPsr4
  187.           (array$paths 
  188.         ); 
  189.       } 
  190.     } elseif (!isset(self::$prefixDirsPsr4[$prefix])) { 
  191.       // Register directories for a new namespace. 
  192.       $length = strlen($prefix); 
  193.       if ('\\' !== $prefix[$length - 1]) { 
  194.         throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator."); 
  195.       } 
  196.       self::$prefixLengthsPsr4[$prefix[0]][$prefix] = $length
  197.       self::$prefixDirsPsr4[$prefix]        = (array$paths
  198.     } elseif ($prepend) { 
  199.       // Prepend directories for an already registered namespace. 
  200.       self::$prefixDirsPsr4[$prefix] = array_merge
  201.         (array$paths
  202.         self::$prefixDirsPsr4[$prefix
  203.       ); 
  204.     } else { 
  205.       // Append directories for an already registered namespace. 
  206.       self::$prefixDirsPsr4[$prefix] = array_merge
  207.         self::$prefixDirsPsr4[$prefix], 
  208.         (array$paths 
  209.       ); 
  210.     } 
  211.   } 
  212.  
  213.   // 注册命名空间别名 
  214.   public static function addNamespaceAlias($namespace$original = ''
  215.   { 
  216.     if (is_array($namespace)) { 
  217.       self::$namespaceAlias = array_merge(self::$namespaceAlias$namespace); 
  218.     } else { 
  219.       self::$namespaceAlias[$namespace] = $original
  220.     } 
  221.   } 
  222.  
  223.   // 注册自动加载机制 
  224.   public static function register($autoload = ''
  225.   { 
  226.     // 注册系统自动加载 
  227.     spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true); 
  228.     // 注册命名空间定义 
  229.     self::addNamespace([ 
  230.       'think'  => LIB_PATH . 'think' . DS, 
  231.       'behavior' => LIB_PATH . 'behavior' . DS, 
  232.       'traits'  => LIB_PATH . 'traits' . DS, 
  233.     ]); 
  234.     // 加载类库映射文件 
  235.     if (is_file(RUNTIME_PATH . 'classmap' . EXT)) { 
  236.       self::addClassMap(__include_file(RUNTIME_PATH . 'classmap' . EXT)); 
  237.     } 
  238.  
  239.     // Composer自动加载支持 
  240.     if (is_dir(VENDOR_PATH . 'composer')) { 
  241.       self::registerComposerLoader(); 
  242.     } 
  243.  
  244.     // 自动加载extend目录 
  245.     self::$fallbackDirsPsr4[] = rtrim(EXTEND_PATH, DS); 
  246.   } 
  247.  
  248.   // 注册composer自动加载 
  249.   private static function registerComposerLoader() 
  250.   { 
  251.     if (is_file(VENDOR_PATH . 'composer/autoload_namespaces.php')) { 
  252.       $map = require VENDOR_PATH . 'composer/autoload_namespaces.php'
  253.       foreach ($map as $namespace => $path) { 
  254.         self::addPsr0($namespace$path); 
  255.       } 
  256.     } 
  257.  
  258.     if (is_file(VENDOR_PATH . 'composer/autoload_psr4.php')) { 
  259.       $map = require VENDOR_PATH . 'composer/autoload_psr4.php'
  260.       foreach ($map as $namespace => $path) { 
  261.         self::addPsr4($namespace$path); 
  262.       } 
  263.     } 
  264.  
  265.     if (is_file(VENDOR_PATH . 'composer/autoload_classmap.php')) { 
  266.       $classMap = require VENDOR_PATH . 'composer/autoload_classmap.php'
  267.       if ($classMap) { 
  268.         self::addClassMap($classMap); 
  269.       } 
  270.     } 
  271.  
  272.     if (is_file(VENDOR_PATH . 'composer/autoload_files.php')) { 
  273.       $includeFiles = require VENDOR_PATH . 'composer/autoload_files.php'
  274.       foreach ($includeFiles as $fileIdentifier => $file) { 
  275.         if (emptyempty(self::$autoloadFiles[$fileIdentifier])) { 
  276.           __require_file($file); 
  277.           self::$autoloadFiles[$fileIdentifier] = true; 
  278.         } 
  279.       } 
  280.     } 
  281.   } 
  282.  
  283.   /** 
  284.    * 导入所需的类库 同java的Import 本函数有缓存功能 
  285.    * @param string $class  类库命名空间字符串 
  286.    * @param string $baseUrl 起始路径 
  287.    * @param string $ext   导入的文件扩展名 
  288.    * @return boolean 
  289.    */ 
  290.   public static function import($class$baseUrl = ''$ext = EXT) 
  291.   { 
  292.     static $_file = []; 
  293.     $key     = $class . $baseUrl
  294.     $class    = str_replace(['.''#'], [DS, '.'], $class); 
  295.     if (isset($_file[$key])) { 
  296.       return true; 
  297.     } 
  298.  
  299.     if (emptyempty($baseUrl)) { 
  300.       list($name$class) = explode(DS, $class, 2); 
  301.  
  302.       if (isset(self::$prefixDirsPsr4[$name . '\\'])) { 
  303.         // 注册的命名空间 
  304.         $baseUrl = self::$prefixDirsPsr4[$name . '\\']; 
  305.       } elseif ('@' == $name) { 
  306.         //加载当前模块应用类库 
  307.         $baseUrl = App::$modulePath
  308.       } elseif (is_dir(EXTEND_PATH . $name)) { 
  309.         $baseUrl = EXTEND_PATH; 
  310.       } else { 
  311.         // 加载其它模块的类库 
  312.         $baseUrl = APP_PATH . $name . DS; 
  313.       } 
  314.     } elseif (substr($baseUrl, -1) != DS) { 
  315.       $baseUrl .= DS; 
  316.     } 
  317.     // 如果类存在 则导入类库文件 
  318.     if (is_array($baseUrl)) { 
  319.       foreach ($baseUrl as $path) { 
  320.         $filename = $path . DS . $class . $ext
  321.         if (is_file($filename)) { 
  322.           break
  323.         } 
  324.       } 
  325.     } else { 
  326.       $filename = $baseUrl . $class . $ext
  327.     } 
  328.  
  329.     if (!emptyempty($filename) && is_file($filename)) { 
  330.       // 开启调试模式Win环境严格区分大小写 
  331.       if (IS_WIN && pathinfo($filename, PATHINFO_FILENAME) != pathinfo(realpath($filename), PATHINFO_FILENAME)) { 
  332.         return false; 
  333.       } 
  334.       __include_file($filename); 
  335.       $_file[$key] = true; 
  336.       return true; 
  337.     } 
  338.     return false; 
  339.   } 
  340.  
  341.   /** 
  342.    * 实例化(分层)模型 
  343.    * @param string $name     Model名称 
  344.    * @param string $layer    业务层名称 
  345.    * @param bool  $appendSuffix 是否添加类名后缀 
  346.    * @param string $common    公共模块名 
  347.    * @return Object 
  348.    * @throws ClassNotFoundException 
  349.    */ 
  350.   public static function model($name = ''$layer = 'model'$appendSuffix = false, $common = 'common'
  351.   { 
  352.     if (isset(self::$instance[$name . $layer])) { 
  353.       return self::$instance[$name . $layer]; 
  354.     } 
  355.     if (strpos($name'/')) { 
  356.       list($module$name) = explode('/'$name, 2); 
  357.     } else { 
  358.       $module = Request::instance()->module(); 
  359.     } 
  360.     $class = self::parseClass($module$layer$name$appendSuffix); 
  361.     if (class_exists($class)) { 
  362.       $model = new $class(); 
  363.     } else { 
  364.       $class = str_replace('\\' . $module . '\\', '\\' . $common . '\\', $class); 
  365.       if (class_exists($class)) { 
  366.         $model = new $class(); 
  367.       } else { 
  368.         throw new ClassNotFoundException('class not exists:' . $class$class); 
  369.       } 
  370.     } 
  371.     self::$instance[$name . $layer] = $model
  372.     return $model
  373.   } 
  374.  
  375.   /** 
  376.    * 实例化(分层)控制器 格式:[模块名/]控制器名 
  377.    * @param string $name     资源地址 
  378.    * @param string $layer    控制层名称 
  379.    * @param bool  $appendSuffix 是否添加类名后缀 
  380.    * @param string $empty    空控制器名称 
  381.    * @return Object|false 
  382.    * @throws ClassNotFoundException 
  383.    */ 
  384.   public static function controller($name$layer = 'controller'$appendSuffix = false, $empty = ''
  385.   { 
  386.     if (strpos($name'/')) { 
  387.       list($module$name) = explode('/'$name); 
  388.     } else { 
  389.       $module = Request::instance()->module(); 
  390.     } 
  391.     $class = self::parseClass($module$layer$name$appendSuffix); 
  392.     if (class_exists($class)) { 
  393.       return new $class(Request::instance()); 
  394.     } elseif ($empty && class_exists($emptyClass = self::parseClass($module$layer$empty$appendSuffix))) { 
  395.       return new $emptyClass(Request::instance()); 
  396.     } 
  397.   } 
  398.  
  399.   /** 
  400.    * 实例化验证类 格式:[模块名/]验证器名 
  401.    * @param string $name     资源地址 
  402.    * @param string $layer    验证层名称 
  403.    * @param bool  $appendSuffix 是否添加类名后缀 
  404.    * @param string $common    公共模块名 
  405.    * @return Object|false 
  406.    * @throws ClassNotFoundException 
  407.    */ 
  408.   public static function validate($name = ''$layer = 'validate'$appendSuffix = false, $common = 'common'
  409.   { 
  410.     $name = $name ?: Config::get('default_validate'); 
  411.     if (emptyempty($name)) { 
  412.       return new Validate; 
  413.     } 
  414.  
  415.     if (isset(self::$instance[$name . $layer])) { 
  416.       return self::$instance[$name . $layer]; 
  417.     } 
  418.     if (strpos($name'/')) { 
  419.       list($module$name) = explode('/'$name); 
  420.     } else { 
  421.       $module = Request::instance()->module(); 
  422.     } 
  423.     $class = self::parseClass($module$layer$name$appendSuffix); 
  424.     if (class_exists($class)) { 
  425.       $validate = new $class
  426.     } else { 
  427.       $class = str_replace('\\' . $module . '\\', '\\' . $common . '\\', $class); 
  428.       if (class_exists($class)) { 
  429.         $validate = new $class
  430.       } else { 
  431.         throw new ClassNotFoundException('class not exists:' . $class$class); 
  432.       } 
  433.     } 
  434.     self::$instance[$name . $layer] = $validate
  435.     return $validate
  436.   } 
  437.  
  438.   /** 
  439.    * 数据库初始化 并取得数据库类实例 
  440.    * @param mixed     $config 数据库配置 
  441.    * @param bool|string  $name 连接标识 true 强制重新连接 
  442.    * @return \think\db\Connection 
  443.    */ 
  444.   public static function db($config = [], $name = false) 
  445.   { 
  446.     return Db::connect($config$name); 
  447.   } 
  448.  
  449.   /** 
  450.    * 远程调用模块的操作方法 参数格式 [模块/控制器/]操作 
  451.    * @param string    $url     调用地址 
  452.    * @param string|array $vars     调用参数 支持字符串和数组 
  453.    * @param string    $layer    要调用的控制层名称 
  454.    * @param bool     $appendSuffix 是否添加类名后缀 
  455.    * @return mixed 
  456.    */ 
  457.   public static function action($url$vars = [], $layer = 'controller'$appendSuffix = false) 
  458.   { 
  459.     $info  = pathinfo($url); 
  460.     $action = $info['basename']; 
  461.     $module = '.' != $info['dirname'] ? $info['dirname'] : Request::instance()->controller(); 
  462.     $class = self::controller($module$layer$appendSuffix); 
  463.     if ($class) { 
  464.       if (is_scalar($vars)) { 
  465.         if (strpos($vars'=')) { 
  466.           parse_str($vars$vars); 
  467.         } else { 
  468.           $vars = [$vars]; 
  469.         } 
  470.       } 
  471.       return App::invokeMethod([$class$action . Config::get('action_suffix')], $vars); 
  472.     } 
  473.   } 
  474.  
  475.   /** 
  476.    * 字符串命名风格转换 
  477.    * type 0 将Java风格转换为C的风格 1 将C风格转换为Java的风格 
  478.    * @param string $name 字符串 
  479.    * @param integer $type 转换类型 
  480.    * @return string 
  481.    */ 
  482.   public static function parseName($name$type = 0) 
  483.   { 
  484.     if ($type) { 
  485.       return ucfirst(preg_replace_callback('/_([a-zA-Z])/'function ($match) { 
  486.         return strtoupper($match[1]); 
  487.       }, $name)); 
  488.     } else { 
  489.       return strtolower(trim(preg_replace("/[A-Z]/""_\\0"$name), "_")); 
  490.     } 
  491.   } 
  492.  
  493.   /** 
  494.    * 解析应用类的类名 
  495.    * @param string $module 模块名 
  496.    * @param string $layer 层名 controller model ... 
  497.    * @param string $name  类名 
  498.    * @param bool  $appendSuffix 
  499.    * @return string 
  500.    */ 
  501.   public static function parseClass($module$layer$name$appendSuffix = false) 
  502.   { 
  503.     $name = str_replace(['/''.'], '\\'$name); 
  504.     $array = explode('\\'$name); 
  505.     $class = self::parseName(array_pop($array), 1) . (App::$suffix || $appendSuffix ? ucfirst($layer) : ''); 
  506.     $path = $array ? implode('\\', $array) . '\\' : ''; 
  507.     return App::$namespace . '\\' . ($module ? $module . '\\' : '') . $layer . '\\' . $path . $class
  508.   } 
  509.  
  510.   /** 
  511.    * 初始化类的实例 
  512.    * @return void 
  513.    */ 
  514.   public static function clearInstance() 
  515.   { 
  516.     self::$instance = []; 
  517.   } 
  518.  
  519. /** 
  520.  * 作用范围隔离 
  521.  * 
  522.  * @param $file 
  523.  * @return mixed 
  524.  */ 
  525. function __include_file($file
  526.   return include $file
  527.  
  528. function __require_file($file
  529.   return require $file
  530. }

Tags: php命名空间 php自动加载

分享到: