PHP命名空间与自动加载机制的基础介绍
前言
include 和 require 是php中引入文件的两个基本方法。在小规模开发中直接使用 include 和 require 没哟什么不妥,但在大型项目中会造成大量的 include 和 require 堆积。这样的代码既不优雅,执行效率也很低,而且维护起来也相当困难。
为了解决这个问题,部分框架会给出一个引入文件的配置清单,在对象初始化的时候把需要的文件引入。但这只是让代码变得更简洁了一些,引入的效果仍然是差强人意。php5 之后,随着 php 面向对象支持的完善,__autoload 函数才真正使得自动加载成为可能。
* include 和 require 功能是一样的,它们的不同在于 include 出错时只会产生警告,而 require 会抛出错误终止脚本。
* include_once 和 include 唯一的区别在于 include_once 会检查文件是否已经引入,如果是则不会重复引入。
=================自动加载==================
实现自动加载最简单的方式就是使用 __autoload 魔术方法。当需要使用的类没有被引入时,这个函数会在php报错前被触发,未定义的类名会被当作参数传入。至于函数具体的逻辑,这需要用户自己去实现。
首先创建一个 autoload.php 来做一个简单的测试:
// 类未定义时,系统自动调用 function __autoload($class) { /* 具体处理逻辑 */ echo $class;// 简单的输出未定义的类名 } new helloworld(); /** * 输出 helloworld 与报错信息 * fatal error: class 'helloworld' not found */
通过这个简单的例子可以发现,在类的实例化过程中,系统所做的工作大致是这样的:
/* 模拟系统实例化过程 */ function instance($class) { // 如果类存在则返回其实例 if (class_exists($class, false)) { return new $class(); } // 查看 autoload 函数是否被用户定义 if (function_exists('__autoload')) { __autoload($class); // 最后一次引入的机会 } // 再次检查类是否存在 if (class_exists($class, false)) { return new $class(); } else { // 系统:我实在没辙了 throw new exception('class not found'); } }
明白了 __autoload 函数的工作原理之后,那就让我们来用它去实现自动加载。
首先创建一个类文件(建议文件名与类名一致),代码如下:
class [classname] { // 对象实例化时输出当前类名 function __construct() { echo '<h1>' . __class__ . '</h1>'; } }
(我这里创建了一个 helloworld 类用作演示)接下来我们就要定义 __autoload 的具体逻辑,使它能够实现自动加载:
function __autoload($class) { // 根据类名确定文件名 $file = $class . '.php'; if (file_exists($file)) { include $file; // 引入php文件 } } new helloworld(); /** * 输出 <h1>helloworld</h1> */
=================命名空间==================
其实命名空间并不是什么新生事物,很多语言(例如c++)早都支持这个特性了。只不过 php 起步比较晚,直到 php 5.3 之后才支持。
命名空间简而言之就是一种标识,它的主要目的是解决命名冲突的问题。
就像在日常生活中,有很多姓名相同的人,如何区分这些人呢?那就需要加上一些额外的标识。
把工作单位当成标识似乎不错,这样就不用担心 “撞名” 的尴尬了。
这里我们来做一个小任务,去介绍百度的ceo李彦宏:
namespace 百度; class 李彦宏 { function __construct() { echo '百度创始人'; } }
↑ 这就是李彦宏的基本资料了,namespace 是他的单位标识,class 是他的姓名。
命名空间通过关键字 namespace 来声明。如果一个文件中包含命名空间,它必须在其它所有代码之前声明命名空间。
new 百度\李彦宏(); // 限定类名 new \百度\李彦宏(); // 完全限定类名
↑ 在一般情况下,无论是向别人介绍 "百度 李彦宏" 还是 "百度公司 李彦宏",他们都能够明白。
在当前命名空间没有声明的情况下,限定类名和完全限定类名是等价的。因为如果不指定空间,则默认为全局(\)。
namespace 谷歌; new 百度\李彦宏(); // 谷歌\百度\李彦宏(实际结果) new \百度\李彦宏(); // 百度\李彦宏(实际结果)
↑ 如果你在谷歌公司向他们的员工介绍李彦宏,一定要指明是 "百度公司的李彦宏"。否则他会认为百度是谷歌的一个部门,而李彦宏只是其中的一位员工而已。
这个例子展示了在命名空间下,使用限定类名和完全限定类名的区别。(完全限定类名 = 当前命名空间 + 限定类名)
/* 导入命名空间 */ use 百度\李彦宏; new 李彦宏(); // 百度\李彦宏(实际结果) /* 设置别名 */ use 百度\李彦宏 as ceo; new ceo(); // 百度\李彦宏(实际结果) /* 任何情况 */ 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 作为它的命名空间(建议文件名与类名保持一致):
namespace os; // 命名空间 class linux // 类名 { function __construct() { echo '<h1>' . __class__ . '</h1>'; } }
接着,在同一个目录下新建一个 php 文件,使用 spl_autoload_register 以函数回调的方式实现自动加载:
spl_autoload_register(function ($class) { // class = os\linux /* 限定类名路径映射 */ $class_map = array( // 限定类名 => 文件路径 'os\\linux' => './linux.php', ); /* 根据类名确定文件名 */ $file = $class_map[$class]; /* 引入相关文件 */ if (file_exists($file)) { include $file; } }); 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:
$class = 'app\view\news\index'; /* *命名空间路径映射 */ $vendor_map = array( 'app' => 'c:\baidu', ); /* 解析类名为文件路径 */ $vendor = substr($class, 0, strpos($class, '\\')); // 取出*命名空间[app] $vendor_dir = $vendor_map[$vendor]; // 文件基目录[c:\baidu] $rel_path = dirname(substr($class, strlen($vendor))); // 相对路径[/view/news] $file_name = basename($class) . '.php'; // 文件名[index.php] /* 输出文件所在路径 */ echo $vendor_dir . $rel_path . directory_separator . $file_name;
通过这个 demo 可以看出限定类名转换为路径的过程。那么现在就让我们用规范的面向对象方式去实现自动加载器吧。
首先我们创建一个文件 index.php,它处于 \app\mvc\view\home 目录中:
namespace app\mvc\view\home; class index { function __construct() { echo '<h1> welcome to home </h1>'; } }
接着我们在创建一个加载类(不需要命名空间),它处于 \ 目录中:
class loader { /* 路径映射 */ public static $vendormap = array( 'app' => __dir__ . directory_separator . 'app', ); /** * 自动加载器 */ public static function autoload($class) { $file = self::findfile($class); if (file_exists($file)) { self::includefile($file); } } /** * 解析文件路径 */ private static function findfile($class) { $vendor = substr($class, 0, strpos($class, '\\')); // *命名空间 $vendordir = self::$vendormap[$vendor]; // 文件基目录 $filepath = substr($class, strlen($vendor)) . '.php'; // 文件相对路径 return strtr($vendordir . $filepath, '\\', directory_separator); // 文件标准路径 } /** * 引入文件 */ private static function includefile($file) { if (is_file($file)) { include $file; } } }
最后,将 loader 类中的 autoload 注册到 spl_autoload_register 函数中:
include 'loader.php'; // 引入加载器 spl_autoload_register('loader::autoload'); // 注册自动加载 new \app\mvc\view\home\index(); // 实例化未引用的类 /** * 输出: <h1> welcome to home </h1> */
示例中的代码其实就是 thinkphp 自动加载器源码的精简版,它是 thinkphp 5 能实现惰性加载的关键。
至此,自动加载的原理已经全部讲完了,如果有兴趣深入了解的话,可以参考下面的 thinkphp 源码。
class loader { protected static $instance = []; // 类名映射 protected static $map = []; // 命名空间别名 protected static $namespacealias = []; // psr-4 private static $prefixlengthspsr4 = []; private static $prefixdirspsr4 = []; private static $fallbackdirspsr4 = []; // psr-0 private static $prefixespsr0 = []; private static $fallbackdirspsr0 = []; // 自动加载的文件 private static $autoloadfiles = []; // 自动加载 public static function autoload($class) { // 检测命名空间别名 if (!empty(self::$namespacealias)) { $namespace = dirname($class); if (isset(self::$namespacealias[$namespace])) { $original = self::$namespacealias[$namespace] . '\\' . basename($class); if (class_exists($original)) { return class_alias($original, $class, false); } } } if ($file = self::findfile($class)) { // win环境严格区分大小写 if (is_win && pathinfo($file, pathinfo_filename) != pathinfo(realpath($file), pathinfo_filename)) { return false; } __include_file($file); return true; } } /** * 查找文件 * @param $class * @return bool */ private static function findfile($class) { if (!empty(self::$map[$class])) { // 类库映射 return self::$map[$class]; } // 查找 psr-4 $logicalpathpsr4 = strtr($class, '\\', ds) . ext; $first = $class[0]; if (isset(self::$prefixlengthspsr4[$first])) { foreach (self::$prefixlengthspsr4[$first] as $prefix => $length) { if (0 === strpos($class, $prefix)) { foreach (self::$prefixdirspsr4[$prefix] as $dir) { if (is_file($file = $dir . ds . substr($logicalpathpsr4, $length))) { return $file; } } } } } // 查找 psr-4 fallback dirs foreach (self::$fallbackdirspsr4 as $dir) { if (is_file($file = $dir . ds . $logicalpathpsr4)) { return $file; } } // 查找 psr-0 if (false !== $pos = strrpos($class, '\\')) { // namespaced class name $logicalpathpsr0 = substr($logicalpathpsr4, 0, $pos + 1) . strtr(substr($logicalpathpsr4, $pos + 1), '_', ds); } else { // pear-like class name $logicalpathpsr0 = strtr($class, '_', ds) . ext; } if (isset(self::$prefixespsr0[$first])) { foreach (self::$prefixespsr0[$first] as $prefix => $dirs) { if (0 === strpos($class, $prefix)) { foreach ($dirs as $dir) { if (is_file($file = $dir . ds . $logicalpathpsr0)) { return $file; } } } } } // 查找 psr-0 fallback dirs foreach (self::$fallbackdirspsr0 as $dir) { if (is_file($file = $dir . ds . $logicalpathpsr0)) { return $file; } } return self::$map[$class] = false; } // 注册classmap public static function addclassmap($class, $map = '') { if (is_array($class)) { self::$map = array_merge(self::$map, $class); } else { self::$map[$class] = $map; } } // 注册命名空间 public static function addnamespace($namespace, $path = '') { if (is_array($namespace)) { foreach ($namespace as $prefix => $paths) { self::addpsr4($prefix . '\\', rtrim($paths, ds), true); } } else { self::addpsr4($namespace . '\\', rtrim($path, ds), true); } } // 添加ps0空间 private static function addpsr0($prefix, $paths, $prepend = false) { if (!$prefix) { if ($prepend) { self::$fallbackdirspsr0 = array_merge( (array) $paths, self::$fallbackdirspsr0 ); } else { self::$fallbackdirspsr0 = array_merge( self::$fallbackdirspsr0, (array) $paths ); } return; } $first = $prefix[0]; if (!isset(self::$prefixespsr0[$first][$prefix])) { self::$prefixespsr0[$first][$prefix] = (array) $paths; return; } if ($prepend) { self::$prefixespsr0[$first][$prefix] = array_merge( (array) $paths, self::$prefixespsr0[$first][$prefix] ); } else { self::$prefixespsr0[$first][$prefix] = array_merge( self::$prefixespsr0[$first][$prefix], (array) $paths ); } } // 添加psr4空间 private static function addpsr4($prefix, $paths, $prepend = false) { if (!$prefix) { // register directories for the root namespace. if ($prepend) { self::$fallbackdirspsr4 = array_merge( (array) $paths, self::$fallbackdirspsr4 ); } else { self::$fallbackdirspsr4 = array_merge( self::$fallbackdirspsr4, (array) $paths ); } } elseif (!isset(self::$prefixdirspsr4[$prefix])) { // register directories for a new namespace. $length = strlen($prefix); if ('\\' !== $prefix[$length - 1]) { throw new \invalidargumentexception("a non-empty psr-4 prefix must end with a namespace separator."); } self::$prefixlengthspsr4[$prefix[0]][$prefix] = $length; self::$prefixdirspsr4[$prefix] = (array) $paths; } elseif ($prepend) { // prepend directories for an already registered namespace. self::$prefixdirspsr4[$prefix] = array_merge( (array) $paths, self::$prefixdirspsr4[$prefix] ); } else { // append directories for an already registered namespace. self::$prefixdirspsr4[$prefix] = array_merge( self::$prefixdirspsr4[$prefix], (array) $paths ); } } // 注册命名空间别名 public static function addnamespacealias($namespace, $original = '') { if (is_array($namespace)) { self::$namespacealias = array_merge(self::$namespacealias, $namespace); } else { self::$namespacealias[$namespace] = $original; } } // 注册自动加载机制 public static function register($autoload = '') { // 注册系统自动加载 spl_autoload_register($autoload ?: 'think\\loader::autoload', true, true); // 注册命名空间定义 self::addnamespace([ 'think' => lib_path . 'think' . ds, 'behavior' => lib_path . 'behavior' . ds, 'traits' => lib_path . 'traits' . ds, ]); // 加载类库映射文件 if (is_file(runtime_path . 'classmap' . ext)) { self::addclassmap(__include_file(runtime_path . 'classmap' . ext)); } // composer自动加载支持 if (is_dir(vendor_path . 'composer')) { self::registercomposerloader(); } // 自动加载extend目录 self::$fallbackdirspsr4[] = rtrim(extend_path, ds); } // 注册composer自动加载 private static function registercomposerloader() { if (is_file(vendor_path . 'composer/autoload_namespaces.php')) { $map = require vendor_path . 'composer/autoload_namespaces.php'; foreach ($map as $namespace => $path) { self::addpsr0($namespace, $path); } } if (is_file(vendor_path . 'composer/autoload_psr4.php')) { $map = require vendor_path . 'composer/autoload_psr4.php'; foreach ($map as $namespace => $path) { self::addpsr4($namespace, $path); } } if (is_file(vendor_path . 'composer/autoload_classmap.php')) { $classmap = require vendor_path . 'composer/autoload_classmap.php'; if ($classmap) { self::addclassmap($classmap); } } if (is_file(vendor_path . 'composer/autoload_files.php')) { $includefiles = require vendor_path . 'composer/autoload_files.php'; foreach ($includefiles as $fileidentifier => $file) { if (empty(self::$autoloadfiles[$fileidentifier])) { __require_file($file); self::$autoloadfiles[$fileidentifier] = true; } } } } /** * 导入所需的类库 同java的import 本函数有缓存功能 * @param string $class 类库命名空间字符串 * @param string $baseurl 起始路径 * @param string $ext 导入的文件扩展名 * @return boolean */ public static function import($class, $baseurl = '', $ext = ext) { static $_file = []; $key = $class . $baseurl; $class = str_replace(['.', '#'], [ds, '.'], $class); if (isset($_file[$key])) { return true; } if (empty($baseurl)) { list($name, $class) = explode(ds, $class, 2); if (isset(self::$prefixdirspsr4[$name . '\\'])) { // 注册的命名空间 $baseurl = self::$prefixdirspsr4[$name . '\\']; } elseif ('@' == $name) { //加载当前模块应用类库 $baseurl = app::$modulepath; } elseif (is_dir(extend_path . $name)) { $baseurl = extend_path; } else { // 加载其它模块的类库 $baseurl = app_path . $name . ds; } } elseif (substr($baseurl, -1) != ds) { $baseurl .= ds; } // 如果类存在 则导入类库文件 if (is_array($baseurl)) { foreach ($baseurl as $path) { $filename = $path . ds . $class . $ext; if (is_file($filename)) { break; } } } else { $filename = $baseurl . $class . $ext; } if (!empty($filename) && is_file($filename)) { // 开启调试模式win环境严格区分大小写 if (is_win && pathinfo($filename, pathinfo_filename) != pathinfo(realpath($filename), pathinfo_filename)) { return false; } __include_file($filename); $_file[$key] = true; return true; } return false; } /** * 实例化(分层)模型 * @param string $name model名称 * @param string $layer 业务层名称 * @param bool $appendsuffix 是否添加类名后缀 * @param string $common 公共模块名 * @return object * @throws classnotfoundexception */ public static function model($name = '', $layer = 'model', $appendsuffix = false, $common = 'common') { if (isset(self::$instance[$name . $layer])) { return self::$instance[$name . $layer]; } if (strpos($name, '/')) { list($module, $name) = explode('/', $name, 2); } else { $module = request::instance()->module(); } $class = self::parseclass($module, $layer, $name, $appendsuffix); if (class_exists($class)) { $model = new $class(); } else { $class = str_replace('\\' . $module . '\\', '\\' . $common . '\\', $class); if (class_exists($class)) { $model = new $class(); } else { throw new classnotfoundexception('class not exists:' . $class, $class); } } self::$instance[$name . $layer] = $model; return $model; } /** * 实例化(分层)控制器 格式:[模块名/]控制器名 * @param string $name 资源地址 * @param string $layer 控制层名称 * @param bool $appendsuffix 是否添加类名后缀 * @param string $empty 空控制器名称 * @return object|false * @throws classnotfoundexception */ public static function controller($name, $layer = 'controller', $appendsuffix = false, $empty = '') { if (strpos($name, '/')) { list($module, $name) = explode('/', $name); } else { $module = request::instance()->module(); } $class = self::parseclass($module, $layer, $name, $appendsuffix); if (class_exists($class)) { return new $class(request::instance()); } elseif ($empty && class_exists($emptyclass = self::parseclass($module, $layer, $empty, $appendsuffix))) { return new $emptyclass(request::instance()); } } /** * 实例化验证类 格式:[模块名/]验证器名 * @param string $name 资源地址 * @param string $layer 验证层名称 * @param bool $appendsuffix 是否添加类名后缀 * @param string $common 公共模块名 * @return object|false * @throws classnotfoundexception */ public static function validate($name = '', $layer = 'validate', $appendsuffix = false, $common = 'common') { $name = $name ?: config::get('default_validate'); if (empty($name)) { return new validate; } if (isset(self::$instance[$name . $layer])) { return self::$instance[$name . $layer]; } if (strpos($name, '/')) { list($module, $name) = explode('/', $name); } else { $module = request::instance()->module(); } $class = self::parseclass($module, $layer, $name, $appendsuffix); if (class_exists($class)) { $validate = new $class; } else { $class = str_replace('\\' . $module . '\\', '\\' . $common . '\\', $class); if (class_exists($class)) { $validate = new $class; } else { throw new classnotfoundexception('class not exists:' . $class, $class); } } self::$instance[$name . $layer] = $validate; return $validate; } /** * 数据库初始化 并取得数据库类实例 * @param mixed $config 数据库配置 * @param bool|string $name 连接标识 true 强制重新连接 * @return \think\db\connection */ public static function db($config = [], $name = false) { return db::connect($config, $name); } /** * 远程调用模块的操作方法 参数格式 [模块/控制器/]操作 * @param string $url 调用地址 * @param string|array $vars 调用参数 支持字符串和数组 * @param string $layer 要调用的控制层名称 * @param bool $appendsuffix 是否添加类名后缀 * @return mixed */ public static function action($url, $vars = [], $layer = 'controller', $appendsuffix = false) { $info = pathinfo($url); $action = $info['basename']; $module = '.' != $info['dirname'] ? $info['dirname'] : request::instance()->controller(); $class = self::controller($module, $layer, $appendsuffix); if ($class) { if (is_scalar($vars)) { if (strpos($vars, '=')) { parse_str($vars, $vars); } else { $vars = [$vars]; } } return app::invokemethod([$class, $action . config::get('action_suffix')], $vars); } } /** * 字符串命名风格转换 * type 0 将java风格转换为c的风格 1 将c风格转换为java的风格 * @param string $name 字符串 * @param integer $type 转换类型 * @return string */ public static function parsename($name, $type = 0) { if ($type) { return ucfirst(preg_replace_callback('/_([a-za-z])/', function ($match) { return strtoupper($match[1]); }, $name)); } else { return strtolower(trim(preg_replace("/[a-z]/", "_\\0", $name), "_")); } } /** * 解析应用类的类名 * @param string $module 模块名 * @param string $layer 层名 controller model ... * @param string $name 类名 * @param bool $appendsuffix * @return string */ public static function parseclass($module, $layer, $name, $appendsuffix = false) { $name = str_replace(['/', '.'], '\\', $name); $array = explode('\\', $name); $class = self::parsename(array_pop($array), 1) . (app::$suffix || $appendsuffix ? ucfirst($layer) : ''); $path = $array ? implode('\\', $array) . '\\' : ''; return app::$namespace . '\\' . ($module ? $module . '\\' : '') . $layer . '\\' . $path . $class; } /** * 初始化类的实例 * @return void */ public static function clearinstance() { self::$instance = []; } } /** * 作用范围隔离 * * @param $file * @return mixed */ function __include_file($file) { return include $file; } function __require_file($file) { return require $file; }
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。