569 lines
16 KiB
PHP
569 lines
16 KiB
PHP
|
<?php
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | ThinkPHP [ WE CAN DO IT JUST THINK ]
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Author: liu21st <liu21st@gmail.com>
|
||
|
// +----------------------------------------------------------------------
|
||
|
|
||
|
namespace think;
|
||
|
|
||
|
use ArrayAccess;
|
||
|
use ArrayIterator;
|
||
|
use Closure;
|
||
|
use Countable;
|
||
|
use InvalidArgumentException;
|
||
|
use IteratorAggregate;
|
||
|
use ReflectionClass;
|
||
|
use ReflectionException;
|
||
|
use ReflectionFunction;
|
||
|
use ReflectionMethod;
|
||
|
use think\exception\ClassNotFoundException;
|
||
|
|
||
|
/**
|
||
|
* @package think
|
||
|
* @property Build $build
|
||
|
* @property Cache $cache
|
||
|
* @property Config $config
|
||
|
* @property Cookie $cookie
|
||
|
* @property Debug $debug
|
||
|
* @property Env $env
|
||
|
* @property Hook $hook
|
||
|
* @property Lang $lang
|
||
|
* @property Middleware $middleware
|
||
|
* @property Request $request
|
||
|
* @property Response $response
|
||
|
* @property Route $route
|
||
|
* @property Session $session
|
||
|
* @property Template $template
|
||
|
* @property Url $url
|
||
|
* @property Validate $validate
|
||
|
* @property View $view
|
||
|
* @property route\RuleName $rule_name
|
||
|
* @property Log $log
|
||
|
*/
|
||
|
class Container implements ArrayAccess, IteratorAggregate, Countable
|
||
|
{
|
||
|
/**
|
||
|
* 容器对象实例
|
||
|
* @var Container
|
||
|
*/
|
||
|
protected static $instance;
|
||
|
|
||
|
/**
|
||
|
* 容器中的对象实例
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $instances = [];
|
||
|
|
||
|
/**
|
||
|
* 容器绑定标识
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $bind = [
|
||
|
'app' => App::class,
|
||
|
'build' => Build::class,
|
||
|
'cache' => Cache::class,
|
||
|
'config' => Config::class,
|
||
|
'cookie' => Cookie::class,
|
||
|
'debug' => Debug::class,
|
||
|
'env' => Env::class,
|
||
|
'hook' => Hook::class,
|
||
|
'lang' => Lang::class,
|
||
|
'log' => Log::class,
|
||
|
'middleware' => Middleware::class,
|
||
|
'request' => Request::class,
|
||
|
'response' => Response::class,
|
||
|
'route' => Route::class,
|
||
|
'session' => Session::class,
|
||
|
'template' => Template::class,
|
||
|
'url' => Url::class,
|
||
|
'validate' => Validate::class,
|
||
|
'view' => View::class,
|
||
|
'rule_name' => route\RuleName::class,
|
||
|
// 接口依赖注入
|
||
|
'think\LoggerInterface' => Log::class,
|
||
|
];
|
||
|
|
||
|
/**
|
||
|
* 容器标识别名
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $name = [];
|
||
|
|
||
|
/**
|
||
|
* 获取当前容器的实例(单例)
|
||
|
* @access public
|
||
|
* @return static
|
||
|
*/
|
||
|
public static function getInstance()
|
||
|
{
|
||
|
if (is_null(static::$instance)) {
|
||
|
static::$instance = new static;
|
||
|
}
|
||
|
|
||
|
return static::$instance;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 设置当前容器的实例
|
||
|
* @access public
|
||
|
* @param object $instance
|
||
|
* @return void
|
||
|
*/
|
||
|
public static function setInstance($instance)
|
||
|
{
|
||
|
static::$instance = $instance;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取容器中的对象实例
|
||
|
* @access public
|
||
|
* @param string $abstract 类名或者标识
|
||
|
* @param array|true $vars 变量
|
||
|
* @param bool $newInstance 是否每次创建新的实例
|
||
|
* @return object
|
||
|
*/
|
||
|
public static function get($abstract, $vars = [], $newInstance = false)
|
||
|
{
|
||
|
return static::getInstance()->make($abstract, $vars, $newInstance);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 绑定一个类、闭包、实例、接口实现到容器
|
||
|
* @access public
|
||
|
* @param string $abstract 类标识、接口
|
||
|
* @param mixed $concrete 要绑定的类、闭包或者实例
|
||
|
* @return Container
|
||
|
*/
|
||
|
public static function set($abstract, $concrete = null)
|
||
|
{
|
||
|
return static::getInstance()->bindTo($abstract, $concrete);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 移除容器中的对象实例
|
||
|
* @access public
|
||
|
* @param string $abstract 类标识、接口
|
||
|
* @return void
|
||
|
*/
|
||
|
public static function remove($abstract)
|
||
|
{
|
||
|
return static::getInstance()->delete($abstract);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 清除容器中的对象实例
|
||
|
* @access public
|
||
|
* @return void
|
||
|
*/
|
||
|
public static function clear()
|
||
|
{
|
||
|
return static::getInstance()->flush();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 绑定一个类、闭包、实例、接口实现到容器
|
||
|
* @access public
|
||
|
* @param string|array $abstract 类标识、接口
|
||
|
* @param mixed $concrete 要绑定的类、闭包或者实例
|
||
|
* @return $this
|
||
|
*/
|
||
|
public function bindTo($abstract, $concrete = null)
|
||
|
{
|
||
|
if (is_array($abstract)) {
|
||
|
$this->bind = array_merge($this->bind, $abstract);
|
||
|
} elseif ($concrete instanceof Closure) {
|
||
|
$this->bind[$abstract] = $concrete;
|
||
|
} elseif (is_object($concrete)) {
|
||
|
if (isset($this->bind[$abstract])) {
|
||
|
$abstract = $this->bind[$abstract];
|
||
|
}
|
||
|
$this->instances[$abstract] = $concrete;
|
||
|
} else {
|
||
|
$this->bind[$abstract] = $concrete;
|
||
|
}
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 绑定一个类实例当容器
|
||
|
* @access public
|
||
|
* @param string $abstract 类名或者标识
|
||
|
* @param object|\Closure $instance 类的实例
|
||
|
* @return $this
|
||
|
*/
|
||
|
public function instance($abstract, $instance)
|
||
|
{
|
||
|
if ($instance instanceof \Closure) {
|
||
|
$this->bind[$abstract] = $instance;
|
||
|
} else {
|
||
|
if (isset($this->bind[$abstract])) {
|
||
|
$abstract = $this->bind[$abstract];
|
||
|
}
|
||
|
|
||
|
$this->instances[$abstract] = $instance;
|
||
|
}
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 判断容器中是否存在类及标识
|
||
|
* @access public
|
||
|
* @param string $abstract 类名或者标识
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function bound($abstract)
|
||
|
{
|
||
|
return isset($this->bind[$abstract]) || isset($this->instances[$abstract]);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 判断容器中是否存在对象实例
|
||
|
* @access public
|
||
|
* @param string $abstract 类名或者标识
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function exists($abstract)
|
||
|
{
|
||
|
if (isset($this->bind[$abstract])) {
|
||
|
$abstract = $this->bind[$abstract];
|
||
|
}
|
||
|
|
||
|
return isset($this->instances[$abstract]);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 判断容器中是否存在类及标识
|
||
|
* @access public
|
||
|
* @param string $name 类名或者标识
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function has($name)
|
||
|
{
|
||
|
return $this->bound($name);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 创建类的实例
|
||
|
* @access public
|
||
|
* @param string $abstract 类名或者标识
|
||
|
* @param array|true $vars 变量
|
||
|
* @param bool $newInstance 是否每次创建新的实例
|
||
|
* @return object
|
||
|
*/
|
||
|
public function make($abstract, $vars = [], $newInstance = false)
|
||
|
{
|
||
|
if (true === $vars) {
|
||
|
// 总是创建新的实例化对象
|
||
|
$newInstance = true;
|
||
|
$vars = [];
|
||
|
}
|
||
|
|
||
|
$abstract = isset($this->name[$abstract]) ? $this->name[$abstract] : $abstract;
|
||
|
|
||
|
if (isset($this->instances[$abstract]) && !$newInstance) {
|
||
|
return $this->instances[$abstract];
|
||
|
}
|
||
|
|
||
|
if (isset($this->bind[$abstract])) {
|
||
|
$concrete = $this->bind[$abstract];
|
||
|
|
||
|
if ($concrete instanceof Closure) {
|
||
|
$object = $this->invokeFunction($concrete, $vars);
|
||
|
} else {
|
||
|
$this->name[$abstract] = $concrete;
|
||
|
return $this->make($concrete, $vars, $newInstance);
|
||
|
}
|
||
|
} else {
|
||
|
$object = $this->invokeClass($abstract, $vars);
|
||
|
}
|
||
|
|
||
|
if (!$newInstance) {
|
||
|
$this->instances[$abstract] = $object;
|
||
|
}
|
||
|
|
||
|
return $object;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 删除容器中的对象实例
|
||
|
* @access public
|
||
|
* @param string|array $abstract 类名或者标识
|
||
|
* @return void
|
||
|
*/
|
||
|
public function delete($abstract)
|
||
|
{
|
||
|
foreach ((array) $abstract as $name) {
|
||
|
$name = isset($this->name[$name]) ? $this->name[$name] : $name;
|
||
|
|
||
|
if (isset($this->instances[$name])) {
|
||
|
unset($this->instances[$name]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取容器中的对象实例
|
||
|
* @access public
|
||
|
* @return array
|
||
|
*/
|
||
|
public function all()
|
||
|
{
|
||
|
return $this->instances;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 清除容器中的对象实例
|
||
|
* @access public
|
||
|
* @return void
|
||
|
*/
|
||
|
public function flush()
|
||
|
{
|
||
|
$this->instances = [];
|
||
|
$this->bind = [];
|
||
|
$this->name = [];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 执行函数或者闭包方法 支持参数调用
|
||
|
* @access public
|
||
|
* @param mixed $function 函数或者闭包
|
||
|
* @param array $vars 参数
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function invokeFunction($function, $vars = [])
|
||
|
{
|
||
|
try {
|
||
|
$reflect = new ReflectionFunction($function);
|
||
|
|
||
|
$args = $this->bindParams($reflect, $vars);
|
||
|
|
||
|
return call_user_func_array($function, $args);
|
||
|
} catch (ReflectionException $e) {
|
||
|
throw new Exception('function not exists: ' . $function . '()');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 调用反射执行类的方法 支持参数绑定
|
||
|
* @access public
|
||
|
* @param mixed $method 方法
|
||
|
* @param array $vars 参数
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function invokeMethod($method, $vars = [])
|
||
|
{
|
||
|
try {
|
||
|
if (is_array($method)) {
|
||
|
$class = is_object($method[0]) ? $method[0] : $this->invokeClass($method[0]);
|
||
|
$reflect = new ReflectionMethod($class, $method[1]);
|
||
|
} else {
|
||
|
// 静态方法
|
||
|
$reflect = new ReflectionMethod($method);
|
||
|
}
|
||
|
|
||
|
$args = $this->bindParams($reflect, $vars);
|
||
|
|
||
|
return $reflect->invokeArgs(isset($class) ? $class : null, $args);
|
||
|
} catch (ReflectionException $e) {
|
||
|
if (is_array($method) && is_object($method[0])) {
|
||
|
$method[0] = get_class($method[0]);
|
||
|
}
|
||
|
|
||
|
throw new Exception('method not exists: ' . (is_array($method) ? $method[0] . '::' . $method[1] : $method) . '()');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 调用反射执行类的方法 支持参数绑定
|
||
|
* @access public
|
||
|
* @param object $instance 对象实例
|
||
|
* @param mixed $reflect 反射类
|
||
|
* @param array $vars 参数
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function invokeReflectMethod($instance, $reflect, $vars = [])
|
||
|
{
|
||
|
$args = $this->bindParams($reflect, $vars);
|
||
|
|
||
|
return $reflect->invokeArgs($instance, $args);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 调用反射执行callable 支持参数绑定
|
||
|
* @access public
|
||
|
* @param mixed $callable
|
||
|
* @param array $vars 参数
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function invoke($callable, $vars = [])
|
||
|
{
|
||
|
if ($callable instanceof Closure) {
|
||
|
return $this->invokeFunction($callable, $vars);
|
||
|
}
|
||
|
|
||
|
return $this->invokeMethod($callable, $vars);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 调用反射执行类的实例化 支持依赖注入
|
||
|
* @access public
|
||
|
* @param string $class 类名
|
||
|
* @param array $vars 参数
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function invokeClass($class, $vars = [])
|
||
|
{
|
||
|
try {
|
||
|
$reflect = new ReflectionClass($class);
|
||
|
|
||
|
if ($reflect->hasMethod('__make')) {
|
||
|
$method = new ReflectionMethod($class, '__make');
|
||
|
|
||
|
if ($method->isPublic() && $method->isStatic()) {
|
||
|
$args = $this->bindParams($method, $vars);
|
||
|
return $method->invokeArgs(null, $args);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$constructor = $reflect->getConstructor();
|
||
|
|
||
|
$args = $constructor ? $this->bindParams($constructor, $vars) : [];
|
||
|
|
||
|
return $reflect->newInstanceArgs($args);
|
||
|
|
||
|
} catch (ReflectionException $e) {
|
||
|
throw new ClassNotFoundException('class not exists: ' . $class, $class);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 绑定参数
|
||
|
* @access protected
|
||
|
* @param \ReflectionMethod|\ReflectionFunction $reflect 反射类
|
||
|
* @param array $vars 参数
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function bindParams($reflect, $vars = [])
|
||
|
{
|
||
|
if ($reflect->getNumberOfParameters() == 0) {
|
||
|
return [];
|
||
|
}
|
||
|
|
||
|
// 判断数组类型 数字数组时按顺序绑定参数
|
||
|
reset($vars);
|
||
|
$type = key($vars) === 0 ? 1 : 0;
|
||
|
$params = $reflect->getParameters();
|
||
|
|
||
|
foreach ($params as $param) {
|
||
|
$name = $param->getName();
|
||
|
$lowerName = Loader::parseName($name);
|
||
|
$class = $param->getClass();
|
||
|
|
||
|
if ($class) {
|
||
|
$args[] = $this->getObjectParam($class->getName(), $vars);
|
||
|
} elseif (1 == $type && !empty($vars)) {
|
||
|
$args[] = array_shift($vars);
|
||
|
} elseif (0 == $type && isset($vars[$name])) {
|
||
|
$args[] = $vars[$name];
|
||
|
} elseif (0 == $type && isset($vars[$lowerName])) {
|
||
|
$args[] = $vars[$lowerName];
|
||
|
} elseif ($param->isDefaultValueAvailable()) {
|
||
|
$args[] = $param->getDefaultValue();
|
||
|
} else {
|
||
|
throw new InvalidArgumentException('method param miss:' . $name);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $args;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取对象类型的参数值
|
||
|
* @access protected
|
||
|
* @param string $className 类名
|
||
|
* @param array $vars 参数
|
||
|
* @return mixed
|
||
|
*/
|
||
|
protected function getObjectParam($className, &$vars)
|
||
|
{
|
||
|
$array = $vars;
|
||
|
$value = array_shift($array);
|
||
|
|
||
|
if ($value instanceof $className) {
|
||
|
$result = $value;
|
||
|
array_shift($vars);
|
||
|
} else {
|
||
|
$result = $this->make($className);
|
||
|
}
|
||
|
|
||
|
return $result;
|
||
|
}
|
||
|
|
||
|
public function __set($name, $value)
|
||
|
{
|
||
|
$this->bindTo($name, $value);
|
||
|
}
|
||
|
|
||
|
public function __get($name)
|
||
|
{
|
||
|
return $this->make($name);
|
||
|
}
|
||
|
|
||
|
public function __isset($name)
|
||
|
{
|
||
|
return $this->bound($name);
|
||
|
}
|
||
|
|
||
|
public function __unset($name)
|
||
|
{
|
||
|
$this->delete($name);
|
||
|
}
|
||
|
|
||
|
public function offsetExists($key)
|
||
|
{
|
||
|
return $this->__isset($key);
|
||
|
}
|
||
|
|
||
|
public function offsetGet($key)
|
||
|
{
|
||
|
return $this->__get($key);
|
||
|
}
|
||
|
|
||
|
public function offsetSet($key, $value)
|
||
|
{
|
||
|
$this->__set($key, $value);
|
||
|
}
|
||
|
|
||
|
public function offsetUnset($key)
|
||
|
{
|
||
|
$this->__unset($key);
|
||
|
}
|
||
|
|
||
|
//Countable
|
||
|
public function count()
|
||
|
{
|
||
|
return count($this->instances);
|
||
|
}
|
||
|
|
||
|
//IteratorAggregate
|
||
|
public function getIterator()
|
||
|
{
|
||
|
return new ArrayIterator($this->instances);
|
||
|
}
|
||
|
|
||
|
public function __debugInfo()
|
||
|
{
|
||
|
$data = get_object_vars($this);
|
||
|
unset($data['instances'], $data['instance']);
|
||
|
|
||
|
return $data;
|
||
|
}
|
||
|
}
|