laravel学习:php写一个简单的ioc服务管理容器

php写一个简单的ioc服务管理容器

原创: 陈晨 CoderStory 2018-01-14

最近学习laravel框架,了解到laravel核心是一个大容器,这个容器负责几乎所有服务组件的实例化以及实例的生命周期管理。这种方式能够很好地对代码进行解耦,使得应用程序的业务代码不必操心服务组件的对象从何而来,当需要一个服务类来完成某个功能的时候,仅需要通过容器解析出该类型的一个实例即可。

最近很焦虑,感觉自己的竞争力越来越弱(现阶段已完全成为一个IT搬砖工),道理大家都懂,但是想要摆脱搬砖工,走向砌墙工需要技术的慢慢积累,千里之行始于足下,那么这个周末就先写一个简单的ioc容器吧 哈哈。。。。

参考了laravel容器的源码,简单分析一番,ioc容器只需实现两个功能,

一、 注册服务(bind)

一般都是在程序刚加载的时候进行服务的注册,只是注册,不进行实例化,在真正用到的时候再进行实例化,这样做减少了资源浪费,做到了按需分配资源。

有些特殊类是单例模式,再注册的时候进行区分

二、实例化服务(make)

从已注册服务的列表中选取要实例化的服务,返回实例对象

=====================================

Container.class.php 代码如下

=====================================

<?php

class Container implements ArrayAccess

{

//注册服务列表

private $_bindings = array();

//实例服务列表

private $_instances = array();

//注册普通服务

public function set($name,$class)

{

$this->bind($name,$class);

}

//注册单例服务

public function setShared($name,$class)

{

$this->bind($name,$class,true);

}

//注册服务

private function bind($name,$class,$shared=false)

{

//卸载服务

$this->remove($name);

//如果是对象直接放入实例服务列表

if(!($class instanceof Closure) && is_object($class))

{

$this->_instances[$name] = $class;

}else{

$this->_bindings[$name]  = array('class' => $class,'shared'=>$shared);

}

}

//获取服务

public function make($name,$params=array()){

//判断服务是否实例化

if (isset($this->_instances[$name])) {

return $this->_instances[$name];

}

//检测是否注册服务

if (!isset($this->_bindings[$name])) {

return null;

}

$concrete = $this->_bindings[$name]['class'];

$obj = null;

//闭包形式注册

if ($concrete instanceof Closure) {

$obj = call_user_func_array($concrete, $params);//通过回调函数调用这个函数

}elseif (is_string($concrete)) {//字符串方式

if (empty($params)) {

$obj= new $concrete;

}else{//直接传入实例对象

//带参数的类的实例化

// $class = new ReflectionClass($concrete);

// $obj = $class->newInstanceArgs($params);

$obj = $concrete;

}

}

//如果是单例模式,则写入_instances列表

if ($this->_bindings[$name]['shared'] == true && $obj) {

$this->_instances[$name] = $obj;

}

return $obj;

}

//检测服务是否存在

public function has($name)

{

return isset($this->_bindings[$name]) or isset($this->_instances[$name]);

}

//卸载服务

public function remove($name)

{

unset($this->_bindings[$name],$this->_instances[$name]);

}

//ArrayAccess接口,检测服务是否存在

public function offsetExists($offset) {

return $this->has($offset);

}

//ArrayAccess接口,以$di[$name]方式获取服务

public function offsetGet($offset) {

return $this->get($offset);

}

//ArrayAccess接口,以$di[$name]=$value方式注册服务,非共享

public function offsetSet($offset, $value) {

return $this->set($offset,$value);

}

//ArrayAccess接口,以unset($di[$name])方式卸载服务

public function offsetUnset($offset) {

return $this->remove($offset);

}

}

?>

=========================================

index.php 测试代码

先声明一个汽车接口类Car,接口有个paiLiang方法

不同品牌的汽车实现Car接口 (假设一个品牌汽车就一种排量)

实现了通过IOC容器管理类的注册和实例化以及依赖注入(通过注入不同的Car实现类,制造不同排量的汽车)

=========================================

<?php

header("Content-Type:text/html;charset=utf8");

include("Container.class.php");

interface Car

{

//排量

public function paiLiang();

}

class Audi implements Car

{

public function paiLiang()

{

return "我是奥迪,排量3.0L";

}

}

class Bmw implements Car

{

public function paiLiang()

{

return "我是宝马,排量2.0T";

}

}

class CarFactory

{

public $paiLiang;

public function __construct(Car $car)

{

$this->paiLiang = $car->paiLiang();

}

}

//实例化容器

$app = new Container();

//注册奥迪和宝马服务

$app->set('audi','Audi');

$app->set('bmw','Bmw');

//注册汽车生产服务

$app->set('carFactory','CarFactory');

//echo $audi->paiLiang();

//获取要生产奥迪车的排量

$newCar = $app->make('carFactory',array($app->make('audi')));

print_r($newCar->paiLiang.'<br>');

$newCar = $app->make('carFactory',array($app->make('bmw')));

print_r($newCar->paiLiang);

输出结果:

我是奥迪,排量3.0L
我是宝马,排量2.0T

上一篇:Alpha版本十天冲刺——Day 1


下一篇:angular ng-bind-html 对src路径失效 解决方案