深入解析Laravel5.5中的包自动发现Package Auto Discovery


Posted in PHP onSeptember 13, 2017

前言

在之前的 Laravel 版本中,安装包通常需要几个步骤,例如添加服务提供器到 app 配置文件并注册相关的 facades。现在,从 Laravel 5.5 开始,Laravel 可以自动检测并注册服务提供器和 facades。

本文不是聚焦于他是怎么用的,而是看看它的源码,是怎么实现Package Auto Discovery的。

composer.json

一切的起源都是来自 composer.json ,在使用 composer 的时候,你可以在 post-autoload-dump 部分指定你想执行的脚本,比如在 Laravel 5.5 的时候,我们可以看到这样的定义:

"scripts": { 
 "post-autoload-dump": [  
  "Illuminate\\Foundation\\ComposerScripts::postAutoloadDump",  
  "@php artisan package:discover"
 ]
}

对于 postAutoloadDump 是很熟悉了,Laravel 之前的版本基本都有,它的工作是清理一些缓存,删除一些旧的文件。

我们的关注重点是@php artisan package:discover ,也就是会执行@php artisan package:discover 这个命令。

这个命令是干嘛的呢?它其实是位于Illuminate\Foundation\Console\PackageDiscoverCommand 中,主要是通过执行Illuminate\Foundation\PackageManifest 的 build() 方法来达到 发现 package 的目的。

而 PackageManifest 早就注册在 Laravel 的 Container 中,那么它可以保证每次在启动 Laravel 的时候都能使用 PackageManifest 的 build() 方法,这个 build() 方法主要的逻辑就是:

找寻 vendor/composer/installed.json 这个文件,这个文件是 composer 自己生成的,记录着每一次的 composer autoload 的 class map。

而此时,Laravel 又将这些内容映射到 extra.laravel 的部分,比如:

"extra": { 
 "laravel": {  
  "providers": [   
  "Barryvdh\\Debugbar\\ServiceProvider"
  ],  
 "aliases": {   
  "Debugbar": "Barryvdh\\Debugbar\\Facade"
  }
 }
}

Laravel 首先将以上内容直接读取下来放到一个 collection 中,然后在去检查下面这个部分的定义,在决定是否需要执行 Package Discover 动作:

"extra": { 
 "laravel": {  
  "dont-discover": [   
  "barryvdh/laravel-debugbar"
  ]
 }
}

如果你不想执行 @php artisan package:discover 的发现效果,可以直接在 dont-discover 的数组里面填上 *。

这样判断完成之后,Laravel 将 collection 中需要发现的 Package 内容保存到一个缓存文件中bootstrap/cache/packages.php:

<?php return array (
 'barryvdh/laravel-debugbar' => array (
 'providers' => array (
  0 => 'Barryvdh\\Debugbar\\ServiceProvider',
 ),
 'aliases' => array (
  'Debugbar' => 'Barryvdh\\Debugbar\\Facade',
 ),
 ),
);

Laravel 怎么拿到 Package 信息?

这个时候,我们又需要看看 Laravel 项目启动的时候,主要启动下面的两个服务:

  • \Illuminate\Foundation\Bootstrap\RegisterFacades
  • \Illuminate\Foundation\Bootstrap\RegisterProvider

而 上面的第一个服务会使用 Illuminate\Foundation\AliasLoader 加载所有的 alias ,在 AliasLoader 中:

// in RegisterFacades::bootstrap()

AliasLoader::getInstance(array_merge(
 $app->make('config')->get('app.aliases', []),
 $app->make(PackageManifest::class)->aliases()
))->register();

你可以看到,它首先读取 app.php 的 aliases 数组,然后与 bootstrap/cache/packages.php 的 packages 数组合并,这个时候,就可以获取到所有的 packages 信息进行发现和自带加载了。这样合并的好处就是,你还是可以直接在 app.php中定义你的 alias 来覆盖自动发现的 package,从而保证你的项目还是可以运行得很流畅。

总结

本文将 Laravel 5.5 Package Auto Discovery 原理和源码梳理的一遍,希望对你了解 Laravel 5.5 Package Auto Discovery 有所帮助。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
mysql+php分页类(已测)
Mar 31 PHP
利用ThinkPHP内置的ThinkAjax实现异步传输技术的实现方法
Dec 19 PHP
php 搜索框提示(自动完成)实例代码
Feb 05 PHP
ThinkPHP模板替换与系统常量及应用实例教程
Aug 22 PHP
PHP加密解密实例分析
Dec 25 PHP
php下载文件超时时间的设置方法
Oct 06 PHP
Laravel学习教程之IOC容器的介绍与用例
Aug 15 PHP
PHP日志LOG类定义与用法示例
Sep 06 PHP
YII2.0框架行为(Behavior)深入详解
Jul 26 PHP
laravel 多图上传及图片的存储例子
Oct 14 PHP
laravel接管Dingo-api和默认的错误处理方式
Oct 25 PHP
thinkPHP框架乐观锁和悲观锁实例分析
Oct 30 PHP
PHP 实现公历日期与农历日期的互转换
Sep 13 #PHP
php生成条形码的图片的实例详解
Sep 13 #PHP
php之可变函数的实例详解
Sep 13 #PHP
PHP实现一个多功能购物网站的案例
Sep 13 #PHP
php之可变变量的实例详解
Sep 12 #PHP
PHP清除缓存的几种方法总结
Sep 12 #PHP
PHP 传输会话curl函数的实例详解
Sep 12 #PHP
You might like
Zerg兵种介绍
2020/03/14 星际争霸
第三节--定义一个类
2006/11/16 PHP
php设计模式 Observer(观察者模式)
2011/06/26 PHP
JpGraph php柱状图使用介绍
2011/08/23 PHP
php文件夹的创建与删除方法
2015/01/24 PHP
Symfony数据校验方法实例分析
2015/01/26 PHP
详解PHP实现异步调用的4种方法
2016/03/14 PHP
JavaScript(JS) 压缩 / 混淆 / 格式化 批处理工具
2010/12/10 Javascript
js 判断脚本加载完毕的代码
2011/07/13 Javascript
document.createElement()用法及注意事项(ff下不兼容)
2013/03/13 Javascript
查询json的数据结构的8种方式简介
2014/03/10 Javascript
javascript正则表达式之分组概念与用法实例
2016/06/16 Javascript
vue的props实现子组件随父组件一起变化
2016/10/27 Javascript
vue二级路由设置方法
2018/02/09 Javascript
浅析vue-router jquery和params传参(接收参数)$router $route的区别
2018/08/03 jQuery
解决Vue.js由于延时显示了{{message}}引用界面的问题
2018/08/25 Javascript
面试题:react和vue的区别分析
2019/04/08 Javascript
React传值 组件传值 之间的关系详解
2019/08/26 Javascript
关于IDEA中的.VUE文件报错 Export declarations are not supported by current JavaScript version
2020/10/17 Javascript
[01:32]2014DOTA2西雅图邀请赛 CIS我们有信心进入正赛
2014/07/08 DOTA
Python的Django框架可适配的各种数据库介绍
2015/07/15 Python
Python之读取TXT文件的方法小结
2018/04/27 Python
pyqt5中QThread在使用时出现重复emit的实例
2019/06/21 Python
python urllib爬虫模块使用解析
2019/09/05 Python
pycharm 添加解释器的方法步骤
2020/08/31 Python
css3高级选择器使用方法
2013/12/02 HTML / CSS
CSS3 清除浮动的方法示例
2018/06/01 HTML / CSS
英国最大的汽车交易网站:Auto Trader UK
2016/09/23 全球购物
意大利奢侈品零售商:ilDuomo Novara
2019/09/11 全球购物
物业经理求职自我评价
2013/09/22 职场文书
2014年师德师风学习材料
2014/05/16 职场文书
计算机毕业生求职信
2014/06/10 职场文书
购房协议书范本(无房产证)
2014/10/07 职场文书
大足石刻导游词
2015/02/02 职场文书
python实现股票历史数据可视化分析案例
2021/06/10 Python
Zabbix对Kafka topic积压数据监控的解决方案
2022/07/07 Servers