Contracts, ServiceContainer, ServiceProvider, Facades关系
概念
-
Contracts 合同,契约,也就是接口,定义一些规则,每个实现此接口的都要实现里面的方法
-
ServiceContainer 实现Contracts,具体的逻辑实现
-
ServiceProvider ServiceContainer的服务提供者,返回ServiceContainer的实例化,供其他地方使用,可以把它加入到app/config的provider中,会被自动注册到容器中
-
Facades 简化ServiceProvider的调用方式,而且可以静态调用ServiceContainer中的方法
实现
Contracts接口可以写或不写,这里就不定义了定义一个ServiceContainer,实现具体的功能
namespace AppHelper;
class MyFoo{
public function add($a, $b)
{
return $a+$b;
}
}
定义一个ServiceProvider供其他地方使用ServiceContain
<?phpnamespace AppProviders;
use AppHelperMyFoo; //要服务的Containeruse IlluminateSupportServiceProvider;
use App;
class MyFooServiceProvider extends ServiceProvider{
public function boot(){}
//注册到容器中
public function register()
{
//可以这么绑定,这需要use App;
App::bind("myfoo",function(){
return new MyFoo();
});
//也可以这么绑定
$this->app->bind("myfoo", function(){
return new MyFoo();
});
}
}
在app/config.php中的providers数组中加入ServiceProvider,让系统自动注册
AppProvidersMyFooServiceProvider::class,
这时候就可以使用了,假设在控制器中使用
public function two($id=null){
//从系统容器中获取实例化对象
//app('myfoo')->add(1,2);
$myfoo = App::make("myfoo");
echo $myfoo->add(1,2);
}
这样太麻烦,还需要用make来获取对象,为了简便,就可以使用门面功能,定义门面MyFooFacade
namespace AppFacades;
use IlluminateSupportFacadesFacade;
class MyFooFacade extends Facade{
protected static function getFacadeAccessor()
{
//这里返回的是ServiceProvider中注册时,定义的字符串
return 'myfoo';
}
}
在控制器里就可以直接调用了
use AppFacadesMyFooFacade;
public function two($id=null){
//从系统容器中获取实例化对象
$myfoo = App::make("myfoo");
echo $myfoo->add(1,2);
//使用门面
echo MyFooFacade::add(4,5);
}
总的来说,自定义了一个类,为了方便在其他别处使用,便可以使用服务提供者和门面
----------------------------华丽的分割线----------------------------
简单的说,“服务容器”就是“服务提供者”提供的服务的容器,“服务容器”通过“服务提供者”来获取服务。
一个laravel应用就是一个“服务容器”,整个框架提供的各种功能由“服务提供者”提供并注入到“服务容器”中。每一个“服务提供者”里register的方法里面使用bind邦定某个对象到一个变量,那个对象就是真正提供服务的对象。
转自:https://segmentfault.com/a/1190000004965752#articleHeader4