418 lines
13 KiB
PHP
418 lines
13 KiB
PHP
<?php
|
|
// +----------------------------------------------------------------------
|
|
// | ThinkPHP [ WE CAN DO IT JUST THINK ]
|
|
// +----------------------------------------------------------------------
|
|
// | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
|
|
// +----------------------------------------------------------------------
|
|
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
|
|
// +----------------------------------------------------------------------
|
|
// | Author: liu21st <liu21st@gmail.com>
|
|
// +----------------------------------------------------------------------
|
|
|
|
namespace think;
|
|
|
|
use think\exception\ClassNotFoundException;
|
|
|
|
class Loader
|
|
{
|
|
/**
|
|
* 类名映射信息
|
|
* @var array
|
|
*/
|
|
protected static $classMap = [];
|
|
|
|
/**
|
|
* 类库别名
|
|
* @var array
|
|
*/
|
|
protected static $classAlias = [];
|
|
|
|
/**
|
|
* PSR-4
|
|
* @var array
|
|
*/
|
|
private static $prefixLengthsPsr4 = [];
|
|
private static $prefixDirsPsr4 = [];
|
|
private static $fallbackDirsPsr4 = [];
|
|
|
|
/**
|
|
* PSR-0
|
|
* @var array
|
|
*/
|
|
private static $prefixesPsr0 = [];
|
|
private static $fallbackDirsPsr0 = [];
|
|
|
|
/**
|
|
* 需要加载的文件
|
|
* @var array
|
|
*/
|
|
private static $files = [];
|
|
|
|
/**
|
|
* Composer安装路径
|
|
* @var string
|
|
*/
|
|
private static $composerPath;
|
|
|
|
// 获取应用根目录
|
|
public static function getRootPath()
|
|
{
|
|
if ('cli' == PHP_SAPI) {
|
|
$scriptName = realpath($_SERVER['argv'][0]);
|
|
} else {
|
|
$scriptName = $_SERVER['SCRIPT_FILENAME'];
|
|
}
|
|
|
|
$path = realpath(dirname($scriptName));
|
|
|
|
if (!is_file($path . DIRECTORY_SEPARATOR . 'think')) {
|
|
$path = dirname($path);
|
|
}
|
|
|
|
return $path . DIRECTORY_SEPARATOR;
|
|
}
|
|
|
|
// 注册自动加载机制
|
|
public static function register($autoload = '')
|
|
{
|
|
// 注册系统自动加载
|
|
spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true);
|
|
|
|
$rootPath = self::getRootPath();
|
|
|
|
self::$composerPath = $rootPath . 'vendor' . DIRECTORY_SEPARATOR . 'composer' . DIRECTORY_SEPARATOR;
|
|
|
|
// Composer自动加载支持
|
|
if (is_dir(self::$composerPath)) {
|
|
if (is_file(self::$composerPath . 'autoload_static.php')) {
|
|
require self::$composerPath . 'autoload_static.php';
|
|
|
|
$declaredClass = get_declared_classes();
|
|
$composerClass = array_pop($declaredClass);
|
|
|
|
foreach (['prefixLengthsPsr4', 'prefixDirsPsr4', 'fallbackDirsPsr4', 'prefixesPsr0', 'fallbackDirsPsr0', 'classMap', 'files'] as $attr) {
|
|
if (property_exists($composerClass, $attr)) {
|
|
self::${$attr} = $composerClass::${$attr};
|
|
}
|
|
}
|
|
} else {
|
|
self::registerComposerLoader(self::$composerPath);
|
|
}
|
|
}
|
|
|
|
// 注册命名空间定义
|
|
self::addNamespace([
|
|
'think' => __DIR__,
|
|
'traits' => dirname(__DIR__) . DIRECTORY_SEPARATOR . 'traits',
|
|
]);
|
|
|
|
// 加载类库映射文件
|
|
if (is_file($rootPath . 'runtime' . DIRECTORY_SEPARATOR . 'classmap.php')) {
|
|
self::addClassMap(__include_file($rootPath . 'runtime' . DIRECTORY_SEPARATOR . 'classmap.php'));
|
|
}
|
|
|
|
// 自动加载extend目录
|
|
self::addAutoLoadDir($rootPath . 'extend');
|
|
}
|
|
|
|
// 自动加载
|
|
public static function autoload($class)
|
|
{
|
|
if (isset(self::$classAlias[$class])) {
|
|
return class_alias(self::$classAlias[$class], $class);
|
|
}
|
|
|
|
if ($file = self::findFile($class)) {
|
|
|
|
// Win环境严格区分大小写
|
|
if (strpos(PHP_OS, 'WIN') !== false && pathinfo($file, PATHINFO_FILENAME) != pathinfo(realpath($file), PATHINFO_FILENAME)) {
|
|
return false;
|
|
}
|
|
|
|
__include_file($file);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 查找文件
|
|
* @access private
|
|
* @param string $class
|
|
* @return string|false
|
|
*/
|
|
private static function findFile($class)
|
|
{
|
|
if (!empty(self::$classMap[$class])) {
|
|
// 类库映射
|
|
return self::$classMap[$class];
|
|
}
|
|
|
|
// 查找 PSR-4
|
|
$logicalPathPsr4 = strtr($class, '\\', DIRECTORY_SEPARATOR) . '.php';
|
|
|
|
$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 . DIRECTORY_SEPARATOR . substr($logicalPathPsr4, $length))) {
|
|
return $file;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// 查找 PSR-4 fallback dirs
|
|
foreach (self::$fallbackDirsPsr4 as $dir) {
|
|
if (is_file($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr4)) {
|
|
return $file;
|
|
}
|
|
}
|
|
|
|
// 查找 PSR-0
|
|
if (false !== $pos = strrpos($class, '\\')) {
|
|
// namespaced class name
|
|
$logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1)
|
|
. strtr(substr($logicalPathPsr4, $pos + 1), '_', DIRECTORY_SEPARATOR);
|
|
} else {
|
|
// PEAR-like class name
|
|
$logicalPathPsr0 = strtr($class, '_', DIRECTORY_SEPARATOR) . '.php';
|
|
}
|
|
|
|
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 . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
|
|
return $file;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// 查找 PSR-0 fallback dirs
|
|
foreach (self::$fallbackDirsPsr0 as $dir) {
|
|
if (is_file($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
|
|
return $file;
|
|
}
|
|
}
|
|
|
|
return self::$classMap[$class] = false;
|
|
}
|
|
|
|
// 注册classmap
|
|
public static function addClassMap($class, $map = '')
|
|
{
|
|
if (is_array($class)) {
|
|
self::$classMap = array_merge(self::$classMap, $class);
|
|
} else {
|
|
self::$classMap[$class] = $map;
|
|
}
|
|
}
|
|
|
|
// 注册命名空间
|
|
public static function addNamespace($namespace, $path = '')
|
|
{
|
|
if (is_array($namespace)) {
|
|
foreach ($namespace as $prefix => $paths) {
|
|
self::addPsr4($prefix . '\\', rtrim($paths, DIRECTORY_SEPARATOR), true);
|
|
}
|
|
} else {
|
|
self::addPsr4($namespace . '\\', rtrim($path, DIRECTORY_SEPARATOR), 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 addAutoLoadDir($path)
|
|
{
|
|
self::$fallbackDirsPsr4[] = $path;
|
|
}
|
|
|
|
// 注册类别名
|
|
public static function addClassAlias($alias, $class = null)
|
|
{
|
|
if (is_array($alias)) {
|
|
self::$classAlias = array_merge(self::$classAlias, $alias);
|
|
} else {
|
|
self::$classAlias[$alias] = $class;
|
|
}
|
|
}
|
|
|
|
// 注册composer自动加载
|
|
public static function registerComposerLoader($composerPath)
|
|
{
|
|
if (is_file($composerPath . 'autoload_namespaces.php')) {
|
|
$map = require $composerPath . 'autoload_namespaces.php';
|
|
foreach ($map as $namespace => $path) {
|
|
self::addPsr0($namespace, $path);
|
|
}
|
|
}
|
|
|
|
if (is_file($composerPath . 'autoload_psr4.php')) {
|
|
$map = require $composerPath . 'autoload_psr4.php';
|
|
foreach ($map as $namespace => $path) {
|
|
self::addPsr4($namespace, $path);
|
|
}
|
|
}
|
|
|
|
if (is_file($composerPath . 'autoload_classmap.php')) {
|
|
$classMap = require $composerPath . 'autoload_classmap.php';
|
|
if ($classMap) {
|
|
self::addClassMap($classMap);
|
|
}
|
|
}
|
|
|
|
if (is_file($composerPath . 'autoload_files.php')) {
|
|
self::$files = require $composerPath . 'autoload_files.php';
|
|
}
|
|
}
|
|
|
|
// 加载composer autofile文件
|
|
public static function loadComposerAutoloadFiles()
|
|
{
|
|
foreach (self::$files as $fileIdentifier => $file) {
|
|
if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) {
|
|
__require_file($file);
|
|
|
|
$GLOBALS['__composer_autoload_files'][$fileIdentifier] = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 字符串命名风格转换
|
|
* type 0 将Java风格转换为C的风格 1 将C风格转换为Java的风格
|
|
* @access public
|
|
* @param string $name 字符串
|
|
* @param integer $type 转换类型
|
|
* @param bool $ucfirst 首字母是否大写(驼峰规则)
|
|
* @return string
|
|
*/
|
|
public static function parseName($name, $type = 0, $ucfirst = true)
|
|
{
|
|
if ($type) {
|
|
$name = preg_replace_callback('/_([a-zA-Z])/', function ($match) {
|
|
return strtoupper($match[1]);
|
|
}, $name);
|
|
return $ucfirst ? ucfirst($name) : lcfirst($name);
|
|
}
|
|
|
|
return strtolower(trim(preg_replace("/[A-Z]/", "_\\0", $name), "_"));
|
|
}
|
|
|
|
/**
|
|
* 创建工厂对象实例
|
|
* @access public
|
|
* @param string $name 工厂类名
|
|
* @param string $namespace 默认命名空间
|
|
* @return mixed
|
|
*/
|
|
public static function factory($name, $namespace = '', ...$args)
|
|
{
|
|
$class = false !== strpos($name, '\\') ? $name : $namespace . ucwords($name);
|
|
|
|
if (class_exists($class)) {
|
|
return Container::getInstance()->invokeClass($class, $args);
|
|
} else {
|
|
throw new ClassNotFoundException('class not exists:' . $class, $class);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 作用范围隔离
|
|
*
|
|
* @param $file
|
|
* @return mixed
|
|
*/
|
|
function __include_file($file)
|
|
{
|
|
return include $file;
|
|
}
|
|
|
|
function __require_file($file)
|
|
{
|
|
return require $file;
|
|
}
|