Loader.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkPHP [ WE CAN DO IT JUST THINK ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
  8. // +----------------------------------------------------------------------
  9. // | Author: liu21st <liu21st@gmail.com>
  10. // +----------------------------------------------------------------------
  11. namespace think;
  12. class Loader
  13. {
  14. /**
  15. * 类名映射信息
  16. * @var array
  17. */
  18. protected static $map = [];
  19. /**
  20. * 类库别名
  21. * @var array
  22. */
  23. protected static $classAlias = [];
  24. /**
  25. * PSR-4
  26. * @var array
  27. */
  28. private static $prefixLengthsPsr4 = [];
  29. private static $prefixDirsPsr4 = [];
  30. private static $fallbackDirsPsr4 = [];
  31. /**
  32. * PSR-0
  33. * @var array
  34. */
  35. private static $prefixesPsr0 = [];
  36. private static $fallbackDirsPsr0 = [];
  37. /**
  38. * 自动加载的文件列表
  39. * @var array
  40. */
  41. private static $autoloadFiles = [];
  42. /**
  43. * Composer安装路径
  44. * @var string
  45. */
  46. private static $composerPath;
  47. // 注册自动加载机制
  48. public static function register($autoload = '')
  49. {
  50. // 注册系统自动加载
  51. spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true);
  52. // 注册命名空间定义
  53. self::addNamespace([
  54. 'think' => __DIR__ . '/',
  55. 'traits' => __DIR__ . '/../traits/',
  56. ]);
  57. $path = dirname($_SERVER['SCRIPT_FILENAME']);
  58. if (is_file('./think')) {
  59. $rootPath = realpath($path) . '/';
  60. } else {
  61. $rootPath = realpath($path . '/../') . '/';
  62. }
  63. // 加载类库映射文件
  64. if (is_file($rootPath . 'runtime/classmap.php')) {
  65. self::addClassMap(__include_file($rootPath . 'runtime/classmap.php'));
  66. }
  67. self::$composerPath = $rootPath . 'vendor/composer/';
  68. // Composer自动加载支持
  69. if (is_dir(self::$composerPath)) {
  70. self::registerComposerLoader(self::$composerPath);
  71. }
  72. // 自动加载extend目录
  73. self::addAutoLoadDir($rootPath . 'extend');
  74. }
  75. // 自动加载
  76. public static function autoload($class)
  77. {
  78. if (isset(self::$classAlias[$class])) {
  79. return class_alias(self::$classAlias[$class], $class);
  80. }
  81. if ($file = self::findFile($class)) {
  82. // Win环境严格区分大小写
  83. if (strpos(PHP_OS, 'WIN') !== false && pathinfo($file, PATHINFO_FILENAME) != pathinfo(realpath($file), PATHINFO_FILENAME)) {
  84. return false;
  85. }
  86. __include_file($file);
  87. return true;
  88. }
  89. }
  90. /**
  91. * 查找文件
  92. * @access private
  93. * @param string $class
  94. * @return string|false
  95. */
  96. private static function findFile($class)
  97. {
  98. if (!empty(self::$map[$class])) {
  99. // 类库映射
  100. return self::$map[$class];
  101. }
  102. // 查找 PSR-4
  103. $logicalPathPsr4 = strtr($class, '\\', DIRECTORY_SEPARATOR) . '.php';
  104. $first = $class[0];
  105. if (isset(self::$prefixLengthsPsr4[$first])) {
  106. foreach (self::$prefixLengthsPsr4[$first] as $prefix => $length) {
  107. if (0 === strpos($class, $prefix)) {
  108. foreach (self::$prefixDirsPsr4[$prefix] as $dir) {
  109. if (is_file($file = $dir . DIRECTORY_SEPARATOR . substr($logicalPathPsr4, $length))) {
  110. return $file;
  111. }
  112. }
  113. }
  114. }
  115. }
  116. // 查找 PSR-4 fallback dirs
  117. foreach (self::$fallbackDirsPsr4 as $dir) {
  118. if (is_file($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr4)) {
  119. return $file;
  120. }
  121. }
  122. // 查找 PSR-0
  123. if (false !== $pos = strrpos($class, '\\')) {
  124. // namespaced class name
  125. $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1)
  126. . strtr(substr($logicalPathPsr4, $pos + 1), '_', DIRECTORY_SEPARATOR);
  127. } else {
  128. // PEAR-like class name
  129. $logicalPathPsr0 = strtr($class, '_', DIRECTORY_SEPARATOR) . '.php';
  130. }
  131. if (isset(self::$prefixesPsr0[$first])) {
  132. foreach (self::$prefixesPsr0[$first] as $prefix => $dirs) {
  133. if (0 === strpos($class, $prefix)) {
  134. foreach ($dirs as $dir) {
  135. if (is_file($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
  136. return $file;
  137. }
  138. }
  139. }
  140. }
  141. }
  142. // 查找 PSR-0 fallback dirs
  143. foreach (self::$fallbackDirsPsr0 as $dir) {
  144. if (is_file($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
  145. return $file;
  146. }
  147. }
  148. return self::$map[$class] = false;
  149. }
  150. // 注册classmap
  151. public static function addClassMap($class, $map = '')
  152. {
  153. if (is_array($class)) {
  154. self::$map = array_merge(self::$map, $class);
  155. } else {
  156. self::$map[$class] = $map;
  157. }
  158. }
  159. // 注册命名空间
  160. public static function addNamespace($namespace, $path = '')
  161. {
  162. if (is_array($namespace)) {
  163. foreach ($namespace as $prefix => $paths) {
  164. self::addPsr4($prefix . '\\', rtrim($paths, DIRECTORY_SEPARATOR), true);
  165. }
  166. } else {
  167. self::addPsr4($namespace . '\\', rtrim($path, DIRECTORY_SEPARATOR), true);
  168. }
  169. }
  170. // 添加Ps0空间
  171. private static function addPsr0($prefix, $paths, $prepend = false)
  172. {
  173. if (!$prefix) {
  174. if ($prepend) {
  175. self::$fallbackDirsPsr0 = array_merge(
  176. (array) $paths,
  177. self::$fallbackDirsPsr0
  178. );
  179. } else {
  180. self::$fallbackDirsPsr0 = array_merge(
  181. self::$fallbackDirsPsr0,
  182. (array) $paths
  183. );
  184. }
  185. return;
  186. }
  187. $first = $prefix[0];
  188. if (!isset(self::$prefixesPsr0[$first][$prefix])) {
  189. self::$prefixesPsr0[$first][$prefix] = (array) $paths;
  190. return;
  191. }
  192. if ($prepend) {
  193. self::$prefixesPsr0[$first][$prefix] = array_merge(
  194. (array) $paths,
  195. self::$prefixesPsr0[$first][$prefix]
  196. );
  197. } else {
  198. self::$prefixesPsr0[$first][$prefix] = array_merge(
  199. self::$prefixesPsr0[$first][$prefix],
  200. (array) $paths
  201. );
  202. }
  203. }
  204. // 添加Psr4空间
  205. private static function addPsr4($prefix, $paths, $prepend = false)
  206. {
  207. if (!$prefix) {
  208. // Register directories for the root namespace.
  209. if ($prepend) {
  210. self::$fallbackDirsPsr4 = array_merge(
  211. (array) $paths,
  212. self::$fallbackDirsPsr4
  213. );
  214. } else {
  215. self::$fallbackDirsPsr4 = array_merge(
  216. self::$fallbackDirsPsr4,
  217. (array) $paths
  218. );
  219. }
  220. } elseif (!isset(self::$prefixDirsPsr4[$prefix])) {
  221. // Register directories for a new namespace.
  222. $length = strlen($prefix);
  223. if ('\\' !== $prefix[$length - 1]) {
  224. throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");
  225. }
  226. self::$prefixLengthsPsr4[$prefix[0]][$prefix] = $length;
  227. self::$prefixDirsPsr4[$prefix] = (array) $paths;
  228. } elseif ($prepend) {
  229. // Prepend directories for an already registered namespace.
  230. self::$prefixDirsPsr4[$prefix] = array_merge(
  231. (array) $paths,
  232. self::$prefixDirsPsr4[$prefix]
  233. );
  234. } else {
  235. // Append directories for an already registered namespace.
  236. self::$prefixDirsPsr4[$prefix] = array_merge(
  237. self::$prefixDirsPsr4[$prefix],
  238. (array) $paths
  239. );
  240. }
  241. }
  242. // 注册自动加载类库目录
  243. public static function addAutoLoadDir($path)
  244. {
  245. self::$fallbackDirsPsr4[] = $path;
  246. }
  247. // 注册类别名
  248. public static function addClassAlias($alias, $class = null)
  249. {
  250. if (is_array($alias)) {
  251. self::$classAlias = array_merge(self::$classAlias, $alias);
  252. } else {
  253. self::$classAlias[$alias] = $class;
  254. }
  255. }
  256. // 注册composer自动加载
  257. public static function registerComposerLoader($composerPath)
  258. {
  259. if (is_file($composerPath . 'autoload_namespaces.php')) {
  260. $map = require $composerPath . 'autoload_namespaces.php';
  261. foreach ($map as $namespace => $path) {
  262. self::addPsr0($namespace, $path);
  263. }
  264. }
  265. if (is_file($composerPath . 'autoload_psr4.php')) {
  266. $map = require $composerPath . 'autoload_psr4.php';
  267. foreach ($map as $namespace => $path) {
  268. self::addPsr4($namespace, $path);
  269. }
  270. }
  271. if (is_file($composerPath . 'autoload_classmap.php')) {
  272. $classMap = require $composerPath . 'autoload_classmap.php';
  273. if ($classMap) {
  274. self::addClassMap($classMap);
  275. }
  276. }
  277. }
  278. // 加载composer autofile文件
  279. public static function loadComposerAutoloadFiles()
  280. {
  281. if (is_file(self::$composerPath . 'autoload_files.php')) {
  282. $includeFiles = require self::$composerPath . 'autoload_files.php';
  283. foreach ($includeFiles as $fileIdentifier => $file) {
  284. if (empty(self::$autoloadFiles[$fileIdentifier])) {
  285. __require_file($file);
  286. self::$autoloadFiles[$fileIdentifier] = true;
  287. }
  288. }
  289. }
  290. }
  291. /**
  292. * 字符串命名风格转换
  293. * type 0 将Java风格转换为C的风格 1 将C风格转换为Java的风格
  294. * @access public
  295. * @param string $name 字符串
  296. * @param integer $type 转换类型
  297. * @param bool $ucfirst 首字母是否大写(驼峰规则)
  298. * @return string
  299. */
  300. public static function parseName($name, $type = 0, $ucfirst = true)
  301. {
  302. if ($type) {
  303. $name = preg_replace_callback('/_([a-zA-Z])/', function ($match) {
  304. return strtoupper($match[1]);
  305. }, $name);
  306. return $ucfirst ? ucfirst($name) : lcfirst($name);
  307. } else {
  308. return strtolower(trim(preg_replace("/[A-Z]/", "_\\0", $name), "_"));
  309. }
  310. }
  311. }
  312. /**
  313. * 作用范围隔离
  314. *
  315. * @param $file
  316. * @return mixed
  317. */
  318. function __include_file($file)
  319. {
  320. return include $file;
  321. }
  322. function __require_file($file)
  323. {
  324. return require $file;
  325. }