您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > php

PHP包管理器Composer实现原理

时间:2019-10-15 13:03:55  来源:  作者:

深入解读PHP包管理器Composer实现原理

 

 

Composer是用php开发的用来管理项目依赖的工具,当你在项目中声明了依赖关系后,composer可以自动帮你下载和安装这些依赖库,并实现自动加载代码。

定义一个composer.json:

{
	"name": "gitlib/composer",
	"require":{
		"predis/predis":"1.1.1"
	}
}

输入命令 composer install,composer会帮我们自动下载predis库,依赖库会默认放在项目的vendor目录下。

├── composer.json
├── composer.lock
├── index.php
└── vendor
 ├── autoload.php
 ├── composer
 │ ├── ClassLoader.php
 │ ├── LICENSE
 │ ├── autoload_classmap.php
 │ ├── autoload_namespaces.php
 │ ├── autoload_psr4.php
 │ ├── autoload_real.php
 │ ├── autoload_static.php
 │ └── installed.json
 └── predis
 └── predis

composer不仅仅帮我们处理依赖,还帮我们实现了自动加载。在vendor目录下有一个autoload.php, 只要在我们的项目中引入这个文件就可以自动加载依赖库。

<?php
require 'vendor/autoload.php';
$client = new PredisClient();
$client->set('foo', 'bar');
$value = $client->get('foo');
echo $value;

可以看到Predis库完全不需要我们手动去加载,只需要require 'vendor/autoload.php',composer的自动加载机制会帮我们找到对应的文件并加载。

对于依赖库,composer帮我们处理好了自动加载, 那对于其他的类库,如何实现自动加载呢?

composer支持四种自动加载的方式:Files/Classmap/PSR-0PSR-4, 其中PSR-4是当前推荐的加载方式。

Files

Files 是最简单的加载方式,这种方式不管加载的文件是否用到始终都会加载,而不是按需加载, 修改项目根目下的composer.json, 加入 “autoload” 项:

{
	"name": "gitlib/composer",
	"require":{
		"predis/predis":"1.1.1"
	},
	"autoload":{
		"files":["Controller/User.php"]
	}
}

files键对应的值是一个数组,数组元素是文件的路径,路径是相对于应用的根目录。加上上述内容后,运行命令:

composer dump-autoload

让composer重建自动加载的信息,composer会把配置值写入与 Files加载方式对应的 verndorcomposerautoload_files.php配置文件中:

<?php
// autoload_files.php @generated by Composer
$vendorDir = dirname(dirname(__FILE__));
$baseDir = dirname($vendorDir);
return array(
 '7efd69bb86214589340b40039fd363f7' => $baseDir . '/Controller/User.php',
);

现在就可以在代码中里调用User类了。

<?php
require 'vendor/autoload.php';
$client = new PredisClient();
$user = new ControllerUser();
$user->login();

Classmap

classmap引用的所有组合,都会在 install/update 过程中生成,并存储到vendor/composer/autoload_classmap.php 文件中。这个 map 是经过扫描指定目录(同样支持直接精确到文件)中所有的 .php 和 .inc 文件里内置的类而得到的。

{
	"name": "gitlib/composer",
	"require":{
		"predis/predis":"1.1.1"
	},
	"autoload":{
		"classmap":["Controller"]
	}
}

Composer会扫描Controller目录下的所有.php.inc文件,存储到vendor/composer/autoload_classmap.php文件中:

<?php
// autoload_classmap.php @generated by Composer
$vendorDir = dirname(dirname(__FILE__));
$baseDir = dirname($vendorDir);
return array(
 'Controller\User' => $baseDir . '/Controller/User.php',
);

PSR-0

PSR-0自动加载规范是已经废弃的标准, 不再做说明。

PSR-4

PSR-4是Composer推荐使用的一种方式(关于PSR规范可参考:PHP标准规范PSR),因为它更易使用并能带来更简洁的目录结构。对于上面的Controller目录我们先改名src:

├── composer.json
├── composer.json.bk
├── composer.lock
├── index.php
├── src
│ └── User.php
└── vendor
 ├── autoload.php
 ├── composer
 └── predis

在composer.json中我们将Controller命名空间和src关联起来:

{
	"name": "gitlib/composer",
	"require":{
		"predis/predis":"1.1.1"
	},
	"autoload":{
		"psr-4": {
			"Controller\":"src/"
		}
	}
}
PSR-4 的命名空间前缀也必须以 \ 结尾,以避免类似前缀间的冲突。

psr-4中的key和value定义了namespace以及其对应的目录映射。按照PSR-4的规则,当试图自动加载”ControllerUser”类的使用,会去寻找”src/User.php”这个文件,此时Controller并不会出现在文件路径中。

自动加载原理

下面我们通过源码分析composer是如何实现自动加载功能。

入口

<?php
require 'vendor/autoload.php';

我们通过require ‘vendor/autoload.php实现自动加载,vendor/autoloaad.php文件引用composer/autoload_real.php

<?php
// autoload.php @generated by Composer
require_once __DIR__ . '/composer/autoload_real.php';
return ComposerAutoloaderInitb84761f57e62a6a534584b91ca213591::getLoader();

autoload_real

autoload_real.php是自动加载引导类,程序主要调用了引导类的静态方法getLoader()

<?php
// autoload_real.php @generated by Composer
class ComposerAutoloaderInitb84761f57e62a6a534584b91ca213591
{
 private static $loader;
 public static function loadClassLoader($class)
 {
 if ('ComposerAutoloadClassLoader' === $class) {
 require __DIR__ . '/ClassLoader.php';
 }
 }
 public static function getLoader()
 {
 // 返回ComposerAutoloadClassLoader单例
 if (null !== self::$loader) {
 return self::$loader;
 }
 // 调用spl_autoload_register加载ComposerAutoloadClassLoader
 spl_autoload_register(array('ComposerAutoloaderInitb84761f57e62a6a534584b91ca213591', 'loadClassLoader'), true, true);
 // 实例化ComposerAutoloadClassLoader类
 self::$loader = $loader = new ComposerAutoloadClassLoader();
 spl_autoload_unregister(array('ComposerAutoloaderInitb84761f57e62a6a534584b91ca213591', 'loadClassLoader'));
 // 静态初始化只支持 PHP5.6 以上版本并且不支持 HHVM 虚拟机
 $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
 if ($useStaticLoader) {
 // 使用 autoload_static 进行静态初始化
 require_once __DIR__ . '/autoload_static.php';
 call_user_func(ComposerAutoloadComposerStaticInitb84761f57e62a6a534584b91ca213591::getInitializer($loader));
 } else {
 // 如果PHP版本低于 5.6 或者使用 HHVM 虚拟机环境,那么就要使用核心类的接口进行初始化
 // PSR0 标准
 $map = require __DIR__ . '/autoload_namespaces.php';
 foreach ($map as $namespace => $path) {
 $loader->set($namespace, $path);
 }
 // PSR4 标准
 $map = require __DIR__ . '/autoload_psr4.php';
 foreach ($map as $namespace => $path) {
 $loader->setPsr4($namespace, $path);
 } 
 // classmap
 $classMap = require __DIR__ . '/autoload_classmap.php';
 if ($classMap) {
 $loader->addClassMap($classMap);
 }
 }
 $loader->register(true);
 // files
 if ($useStaticLoader) {
 $includeFiles = ComposerAutoloadComposerStaticInitb84761f57e62a6a534584b91ca213591::$files;
 } else {
 $includeFiles = require __DIR__ . '/autoload_files.php';
 }
 // files定义的文件,直接require就行了
 foreach ($includeFiles as $fileIdentifier => $file) {
 composerRequireb84761f57e62a6a534584b91ca213591($fileIdentifier, $file);
 }
 return $loader;
 }
}

autoload_static

<?php
// autoload_static.php @generated by Composer
namespace ComposerAutoload;
class ComposerStaticInitb84761f57e62a6a534584b91ca213591
{
 public static $files = array (
 '7efd69bb86214589340b40039fd363f7' => __DIR__ . '/../..' . '/Controller/User.php',
 );
 public static $prefixLengthsPsr4 = array (
 'P' => 
 array (
 'Predis\' => 7,
 ),
 'C' => 
 array (
 'Controller\' => 11,
 ),
 );
 public static $prefixDirsPsr4 = array (
 'Predis\' => 
 array (
 0 => __DIR__ . '/..' . '/predis/predis/src',
 ),
 'Controller\' => 
 array (
 0 => __DIR__ . '/../..' . '/src',
 ),
 );
 public static function getInitializer(ClassLoader $loader)
 {
 return Closure::bind(function () use ($loader) {
 $loader->prefixLengthsPsr4 = ComposerStaticInitb84761f57e62a6a534584b91ca213591::$prefixLengthsPsr4;
 $loader->prefixDirsPsr4 = ComposerStaticInitb84761f57e62a6a534584b91ca213591::$prefixDirsPsr4;
 }, null, ClassLoader::class);
 }
}

静态初始化类的核心就是 getInitializer() 函数,它将自己类中的顶级命名空间映射给了 ClassLoader 类。

PSR4 标准顶级命名空间映射用了两个数组,第一个是用命名空间第一个字母作为前缀索引,然后是 顶级命名空间,但是最终并不是文件路径,而是 顶级命名空间的长度。为什么呢?

因为 PSR4 标准是用顶级命名空间目录替换顶级命名空间,所以获得顶级命名空间的长度很重要。

ClassLoader

public function register($prepend = false)
{
 spl_autoload_register(array($this, 'loadClass'), true, $prepend);
}
public function loadClass($class)
{
 if ($file = $this->findFile($class)) {
 includeFile($file);
 return true;
 }
}
/**
* Finds the path to the file where the class is defined.
*
* @param string $class The name of the class
*
* @return string|false The path if found, false otherwise
*/
public function findFile($class)
{
 // class map lookup
 if (isset($this->classMap[$class])) {
 return $this->classMap[$class];
 }
 if ($this->classMapAuthoritative || isset($this->missingClasses[$class])) {
 return false;
 }
 if (null !== $this->apcuPrefix) {
 $file = apcu_fetch($this->apcuPrefix.$class, $hit);
 if ($hit) {
 return $file;
 }
 }
 $file = $this->findFileWithExtension($class, '.php');
 // Search for Hack files if we are running on HHVM
 if (false === $file && defined('HHVM_VERSION')) {
 $file = $this->findFileWithExtension($class, '.hh');
 }
 if (null !== $this->apcuPrefix) {
 apcu_add($this->apcuPrefix.$class, $file);
 }
 if (false === $file) {
 // Remember that this class does not exist.
 $this->missingClasses[$class] = true;
 }
 return $file;
}
function includeFile($file)
{
 include $file;
}

ClassLoader 的 register() 函数将 loadClass() 函数注册到 PHP 的 SPL 函数堆栈中,每当 PHP 遇到不认识的命名空间时就会调用函数堆栈的每个函数,直到加载命名空间成功。所以 loadClass() 函数就是自动加载的关键了。



Tags:PHP Composer   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
PHP开发免不了要使用Composer,使用Composer有着明显的好处。一是可以很好的解决第三方包的相互依赖,二是可以方便进行代码的重用。那如何制作自己的Composer包呢?制作Composer...【详细内容】
2020-09-21  Tags: PHP Composer  点击:(114)  评论:(0)  加入收藏
安装#进入安装目录cd /usr/local/bin#下载并安装sudo curl -s https://getcomposer.org/installer | sudo php#添加执行权限sudo chmod a+x composer.phar#加入全局命令mv...【详细内容】
2020-05-25  Tags: PHP Composer  点击:(77)  评论:(0)  加入收藏
Composer是用PHP开发的用来管理项目依赖的工具,当你在项目中声明了依赖关系后,composer可以自动帮你下载和安装这些依赖库,并实现自动加载代码。定义一个composer.json:{ "n...【详细内容】
2019-10-15  Tags: PHP Composer  点击:(105)  评论:(0)  加入收藏
▌简易百科推荐
序言:前段时间织梦因为版权的问题在网上闹得沸沸扬扬,也提醒了众多开发者选择cms上应该谨慎使用,今天给大家展示一款自己搭建的内容管理系统,不用担心版权的问题,而且非常容易维...【详细内容】
2021-11-30  小程序软件开发    Tags:管理系统   点击:(34)  评论:(0)  加入收藏
准备安装包(PHP: Hypertext Preprocessor)下载安装包以及组件wget https://www.php.net/distributions/php-8.0.0.tar.bz2wget https://github.com/phpredis/phpredis/archive...【详细内容】
2021-11-09  mimic96    Tags:PHP   点击:(40)  评论:(0)  加入收藏
golang context 很好用,就使用php实现了github地址 : https://github.com/qq1060656096/php-go-context context使用闭坑指南1. 将一个Context参数作为第一个参数传递给传入和...【详细内容】
2021-11-05  1060656096    Tags:PHP   点击:(41)  评论:(0)  加入收藏
一段数组为例:$list = array:4 [ 0 => array:7 [ "id" => 56 "mer_id" => 7 "order_id" => "wx163265961408769974" "is_postage" => 0 "store_name" => "奇...【详细内容】
2021-09-29  七七小影视    Tags:PHP   点击:(65)  评论:(0)  加入收藏
利用JS的CryptoJS 3.x和PHP的openssl_encrypt,openssl_decrypt实现AES对称加密解密,由于需要两种语言对同一字符串的操作,而CryptoJS 的默认加密方式为“aes-256-cbc”,PHP端也...【详细内容】
2021-09-16  李老师tome    Tags:对称加密   点击:(79)  评论:(0)  加入收藏
1、checkdate()验证格利高里日期即:日期是否存在。checkdate(month,day,year);month必需。一个从 1 到 12 的数字,规定月。day必需。一个从 1 到 31 的数字,规定日。year必需。...【详细内容】
2021-08-31  七七小影视    Tags:时间函数   点击:(80)  评论:(0)  加入收藏
对于各类开发语言来说,整数都有一个最大的位数,如果超过位数就无法显示或者操作了。其实,这也是一种精度越界之后产生的精度丢失问题。在我们的 PHP 代码中,最大的整数非常大,我...【详细内容】
2021-08-26  硬核项目经理    Tags:PHP   点击:(83)  评论:(0)  加入收藏
遵从所有教材以及各类数据结构相关的书书籍,我们先从线性表开始入门。今天这篇文章更偏概念,是关于有线性表的一个知识点的汇总。上文说过,物理结构是用于确定数据以何种方式存...【详细内容】
2021-07-19  硬核项目经理    Tags:线性表   点击:(94)  评论:(0)  加入收藏
一、开启IIS全部功能。二、部署PHP1.官网下载并解压PHP: https://windows.php.net/downloads/releases/2.将php.ini-development文件改为php.ini3.修改php.ini(1)去掉注释,并修...【详细内容】
2021-07-15  炘蓝火诗  今日头条  Tags:PHP环境   点击:(129)  评论:(0)  加入收藏
一、环境说明本文中使用本地VM虚机部署测试。OS:CentOS Linux release 7.8.2003 (Core)虚机配置:2核CPU、4G内存①系统为CentOS 7.8 x64最小化安装,部署前已完成系统初始化、...【详细内容】
2021-06-25  IT运维笔记  今日头条  Tags:PHP8.0.7   点击:(141)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条