laravel 服务容器实现原理

前言

通过实现laravel 框架功能,以便深入理解laravel框架的先进思想。

什么是服务容器

服务容器是用来管理类依赖与运行依赖注入的工具。Laravel框架中就是使用服务容器来实现 ** 控制反转 ** 和 ** 依赖注入 **。

什么是控制反转(IoC)和依赖注入(DI)

控制反转(IoC) 就是说把创建对象的** 控制权 进行转移,以前创建对象的主动权和创建时机是由自己把控的,而现在这种权力转移到第三方,也就是 Laravel ** 中的容器。

依赖注入(DI)则是帮助容器实现在运行中动态的为对象提供提依赖的资源。

概念容易不太容易让人理解,举个栗子:

//我们构建一个人的类和一个狗的类
 class People{    public $dog = null;    public function __construct()
    {
        $this->dog = new Dog();
    }    public function putDog(){        return $this->dog->dogCall();
    }

}class Dog{    public function dogCall(){        return ‘汪汪汪‘;
    }
}
 ``` 
这个人在遛狗,突然遇到了死对头,他于是放狗咬人

$people = new People();
$people->putDog();
在这个操作中,people类要执行 putDog() 这个方法,需要依赖Dog类,一般我们像上面一样,在people中利用构造函数来添加这个Dog依赖。如果使用控制反转 依赖注入则是这个样子
class People
{
public $dog = null;

public function __construct(Dog $Dog){
    $this->dog = $dog;
}public function putDog(){    return $this->dog->dogCall();
}

}

People类通过构造参数声明自己需要的 依赖类,由容器自动注入。这样就实现了程序的有效解耦,好处在这就不多说了。## Laravel容器依赖注入的实现###### 实现原理需要了解的知识点:> 闭包(匿名函数):匿名函数(Anonymous functions),也叫闭包函数(closures),允许 临时创建一个没有指定名称的函数> 反射:PHP 5 以上版本具有完整的反射 API,添加了对类、接口、函数、方法和扩展进行反向工程的能力。 此外,反射 API 提供了方法来取出函数、类和方法中的文档注释###### 理解了闭包和反射的基本用法我们来看Laravel中是怎么实现容器的,下面代码是我对laravel框架容器部分代码的简化核心版:

lass Container
{
/**
* 容器绑定,用来装提供的实例或者 提供实例的回调函数
* @var array
*/
public $building = [];

/**
 * 注册一个绑定到容器
 */public function bind($abstract, $concrete = null, $shared = false){    if(is_null($concrete)){
        $concrete = $abstract;
    }    if(!$concrete instanceOf Closure){
        $concrete = $this->getClosure($abstract, $concrete);
    }

    $this->building[$abstract] =  compact("concrete", "shared");
}//注册一个共享的绑定 单例public function singleton($abstract, $concrete, $shared = true){
    $this->bind($abstract, $concrete, $shared);
}/**
 * 默认生成实例的回调闭包
 *
 * @param $abstract
 * @param $concrete
 * @return Closure
 */public function getClosure($abstract, $concrete){    return function($c) use($abstract, $concrete){
        $method = ($abstract == $concrete)? ‘build‘ : ‘make‘;        return $c->$method($concrete);
    };
}/**
 * 生成实例 
 */public function make($abstract){
    $concrete = $this->getConcrete($abstract);    if($this->isBuildable($concrete, $abstract)){
        $object = $this->build($concrete);
    }else{
        $object = $this->make($concrete);
    }    return $object;
}/**
 * 获取绑定的回调函数
 */public function getConcrete($abstract){    if(! isset($this->building[$abstract])){        return $abstract;
    }    return $this->building[$abstract][‘concrete‘];
}/**
 * 判断 是否 可以创建服务实体
 */public function isBuildable($concrete, $abstract){    return $concrete === $abstract || $concrete instanceof Closure;
}/**
 * 根据实例具体名称实例具体对象
 */public function build($concrete){    if($concrete instanceof Closure){        return $concrete($this);
    }    //创建反射对象
    $reflector = new ReflectionClass($concrete);    if( ! $reflector->isInstantiable()){        //抛出异常
        throw new \Exception(‘无法实例化‘);
    }

    $constructor = $reflector->getConstructor();    if(is_null($constructor)){        return new $concrete;
    }

    $dependencies = $constructor->getParameters();
    $instance = $this->getDependencies($dependencies);    return $reflector->newInstanceArgs($instance);

}//通过反射解决参数依赖public function getDependencies(array $dependencies){
    $results = [];    foreach( $dependencies as $dependency ){
        $results[] = is_null($dependency->getClass())
            ?$this->resolvedNonClass($dependency)
            :$this->resolvedClass($dependency);
    }    return $results;
}//解决一个没有类型提示依赖public function resolvedNonClass(ReflectionParameter $parameter){    if($parameter->isDefaultValueAvailable()){        return $parameter->getDefaultValue();
    }    throw new \Exception(‘出错‘);

}//通过容器解决依赖public function resolvedClass(ReflectionParameter $parameter){    return $this->make($parameter->getClass()->name);

}

}
```

容器的工作流程

接着上面遛狗的例子:
//实例化容器类 $app = new Container(); //向容器中填充Dog $app->bind(‘Dog‘,‘App\Dog‘); //填充People $app->bind(‘People‘, ‘App\People‘); //通过容器实现依赖注入,完成类的实例化; $people = $app->make(‘People‘); //调用方法 echo $people->putDog();
上面示例中我们先实例化容器类,然后使用bind()方法 绑定接口和 生成相应的实例的闭包函数。然后使用make() 函数生成实例对象,在make()中会调用 isBuildable($concrete, $abstract) 来判断 给定的服务实体($concrete参数)是否可以创建,可以创建 就会调用 build($concrete) 函数 ,build($concrete)函数会判断传的参数是 是** 闭包 还是 具体类名 **,如果是闭包则直接运行,如果是具体类名的话,则通过反射获取该类的构造函数所需的依赖,完成实例化。

** 重点理解 下面这几个函数中 反射的用法,应该就很好理解了 **
build($concrete) getDependencies(array $dependencies) resolvedNonClass(ReflectionParameter $parameter) resolvedClass(ReflectionParameter $parameter)

时间: 2024-10-27 08:14:51

laravel 服务容器实现原理的相关文章

Laravel 服务容器 IoC(控制反转) 和 DI(依赖注入)

Laravel 服务容器 IoC(控制反转) 和 DI(依赖注入) IoC 容器, laravel 的核心 Laravel 的核心就是一个 IoC 容器,根据文档,称其为“服务容器”,顾名思义,该容器提供了整个框架中需要的一系列服务.作为初学者,很多人会在这一个概念上犯难,因此,我打算从一些基础的内容开始讲解,通过理解面向对象开发中依赖的产生和解决方法,来逐渐揭开“依赖注入”的面纱,逐渐理解这一神奇的设计理念. 本文一大半内容都是通过举例来让读者去理解什么是 IoC(控制反转) 和 DI(依赖注

laravel服务容器

\public\index.php 1 <?php 2 define('LARAVEL_START', microtime(true)); 3 //注册自动加载文件 4 require __DIR__.'/../vendor/autoload.php'; 5 /** 6 * 服务容器的生成 7 * 主要实现了服务容器的实例化和基本注册 8 *包括服务容器本身的注册,基础服务提供者注册,核心类别名注册和基本路径注册 9 * 10 */ 11 $app = require_once __DIR__.

laravel 服务容器的由来 代码展示

1 <?php 2 /** 3 * 目的:代码的完善来说明从 基础类的调用到 工厂类的使用 再到容器的出现的原因 4 * (首先要明白工厂类和容器的关系 可以理解:容器就是工厂类的升级版(为了解决类的依赖)) 5 * 如果不明白工厂类的往下看看,对比一下这几个例子,相信你就明白了. 6 * 下面举个例子: 7 * 简单模拟一个超人 8 * 1.一个超人 具备超人的能力(就是超人技能:如飞行 射击 扔炸弹 暴力攻击等能力) 9 * 2.从面向对象设计:首先分析应该分为超人类(Superman) 和

laravel 服务容器的用法

建立一个服务 <?php namespace App\Services; class FooService { public function __construct(){ } public function doSomething(){ echo "some"; } } 绑定服务 $this->app->bind('\App\Services\FooService'); 在控制器里调用$fooService = app()->make('\App\Servic

laravel 学习笔记 —— 神奇的服务容器

转载自:https://www.insp.top/learn-laravel-container 容器,字面上理解就是装东西的东西.常见的变量.对象属性等都可以算是容器.一个容器能够装什么,全部取决于你对该容器的定义.当然,有这样一种容器,它存放的不是文本.数值,而是对象.对象的描述(类.接口)或者是提供对象的回调,通过这种容器,我们得以实现许多高级的功能,其中最常提到的,就是 “解耦” .“依赖注入(DI)”.本文就从这里开始. IoC 容器, laravel 的核心 Laravel 的核心就

laravel框架总结(四) -- 服务容器

1.依赖 我们定义两个类:class Supperman 和 class Power,现在我们要使用Supperman ,而Supperman 依赖了Power class Supperman { private $power; public function __construct(){ $this->power = new Power; } } 一旦Power发生了变化,Supperman 不得不修改,这种就叫耦合程度太高,所以面临的问题是解耦,就需要用到控制反转. 2.依赖注入 只要不是由

Laravel服务/服务提供者/容器/契约和门面

1.服务是什么? 服务是提供了一些功能的类,比如发送邮件,写日志. 2.Laravel服务提供者是什么? 服务提供者中指明了这个提供者可以提供哪些服务(注册服务),以及服务注册后默认调用一些方法(boot). 3.能否不用服务提供者,直接调用服务? 可以,依靠Laravel的依赖注入,你可以方便的调用任何一个服务,而不用服务提供者. 4.既然3成立,那么服务提供者有什么存在的必要? a.利用服务提供者的boot方法,可以做一些全局性的设置,如在boot中对view的composerb.服务提供者

laravel框架容器管理的一些要点

本文面向php语言的laravel框架的用户,介绍一些laravel框架里面容器管理方面的使用要点.文章很长,但是内容应该很有用,希望有需要的朋友能看到.php经验有限,不到位的地方,欢迎帮忙指正. 1. laravel容器基本认识 laravel框架是有一个容器框架,框架应用程序的实例就是一个超大的容器,这个实例在bootstrap/app.php内进行初始化: 这个文件在每一次请求到达laravel框架都会执行,所创建的$app即是laravel框架的应用程序实例,它在整个请求生命周期都是唯

[php]laravel框架容器管理的一些要点

本文面向php语言的laravel框架的用户,介绍一些laravel框架里面容器管理方面的使用要点.文章很长,但是内容应该很有用,希望有需要的朋友能看到.php经验有限,不到位的地方,欢迎帮忙指正. 1. laravel容器基本认识 laravel框架是有一个容器框架,框架应用程序的实例就是一个超大的容器,这个实例在bootstrap/app.php内进行初始化: 这个文件在每一次请求到达laravel框架都会执行,所创建的$app即是laravel框架的应用程序实例,它在整个请求生命周期都是唯