使用ioc(控制反转)容器的目的就是为了降低DI(依赖注入);
class Container
{
protected $binds;
protected $instances;
public function bind($abstract, $concrete)
{
if ($concrete instanceof Closure) { #如果$concrete是一个闭包即匿名函数的话两个参数以键值对的形式存入$binds数组
$this->binds[$abstract] = $concrete;
} else {
$this->instances[$abstract] = $concrete;
}
}
public function make($abstract, $parameters = [])
{
if (isset($this->instances[$abstract])) {
return $this->instances[$abstract];
}
#将当前对象,即Container放入参数列表的开头部分
array_unshift($parameters, $this);
#调用对应$abstract的$concrete,将此闭包的返回值返回。
return call_user_func_array($this->binds[$abstract], $parameters);
}
}
# 向该超级工厂添加超人的生产脚本
$container->bind('superman', function($container, $moduleName) {
return new Superman($container->make($moduleName));
});
# 向该超级工厂添加超能力模组 的生产脚本
$container->bind('xpower', function($container) {
return new XPower;
});
# 开始启动生产,
$superman_1 = $container->make('superman', 'xpower');
当我们调用$superman_1 = $container->make('superman', 'xpower');
方法时,会调用之前注册在$binds
数组上的键名为$superman
的值,且传递参数为[当前对象,xpower]
,键名为$superman
的值即为
function($container, $moduleName) {
return new Superman($container->make($moduleName));
}
执行此函数,返回值为superman对象,其参数为$container->make($moduleName)
,$moduleName即为xpower
,此时会递归的调用make方法,即又执行了之前绑定在$binds
数组中的键为xpower的值,且参数为[当前对象]
,键为xpower的值即
function($container) {
return new XPower;
}
此函数返回XPower对象。new superman时得到参数,也返回superman对象,$superman_1 = $container->make('superman', 'xpower');
以上为这行代码的执行过程,