依赖注入:在实际需要时拉出所需的组件

Gri*_*lij 18 php design-patterns dependency-injection

DI背后的要点是减轻一个类创建和准备它依赖的对象并将它们推入.这听起来很合理,但有时一个类不需要所有的对象,它们被推入它来执行它的功能.这背后的原因是"早期返回"发生在无效的用户输入或之前所需对象之一抛出的异常,或者在代码块运行之前实例化对象所必需的某个值不可用.

更实际的例子:

  • 注入永远不会使用的数据库连接对象,因为用户数据未通过验证(前提是没有使用触发器来验证此数据)
  • 注入收集输入的类似excel的对象(例如PHPExcel)(因为整个库被拉入并且从未使用过,所以加载和实例化很重,因为验证会在写入发生之前抛出异常)
  • 在类中确定的变量值,但在运行时不是注入器; 例如,路由组件,用于确定应根据用户输入调用的控制器(或命令)类和方法
  • 虽然这可能是一个设计问题,但是一个实质性的服务类,这取决于很多组件,但每个请求只使用1/3(原因,为什么我倾向于使用命令类而不是控制器)

因此,推动所有必要组件的方式与某些组件的创建和从未使用的方式相矛盾的是"延迟加载",这有点不实用且影响性能.就PHP而言 - 加载,解析和编译更多文件.如果被推入的对象具有自己的依赖关系,则这尤其痛苦.

我看到它有3种方法,其中2种听起来不太好:

  • 注射工厂
  • 注入喷射器(反模式)
  • 注入一些外部函数,一旦到达相关点就从类中调用(smtg,如"数据验证完成后检索PHPExcel实例"); 由于其灵活性,这是我倾向于使用的

问题是处理这种情况的最佳方法是什么/你们使用什么?

更新:@GordonM这里有3种方法的例子:

//inject factory example
interface IFactory{
    function factory();
}
class Bartender{
    protected $_factory;

    public function __construct(IFactory $f){
        $this->_factory = $f;
    }
    public function order($data){
        //validating $data
        //... return or throw exception
        //validation passed, order must be saved
        $db = $this->_factory->factory(); //! factory instance * num necessary components
        $db->insert('orders', $data);
        //...
    }
}

/*
inject provider example
assuming that the provider prepares necessary objects
(i.e. injects their dependencies as well)
*/
interface IProvider{
    function get($uid);
}
class Router{
    protected $_provider;

    public function __construct(IProvider $p){
        $this->_provider = $p;
    }
    public function route($str){
        //... match $str against routes to resolve class and method
        $inst = $this->_provider->get($class);
        //...
    }
}

//inject callback (old fashion way)
class MyProvider{
    protected $_db;
    public function getDb(){
        $this->_db = $this->_db ? $this->_db : new mysqli();
        return $this->_db;
    }
}
class Bartender{
    protected $_db;

    public function __construct(array $callback){
        $this->_db = $callback;
    }
    public function order($data){
        //validating $data
        //... return or throw exception
        //validation passed, order must be saved
        $db = call_user_func_array($this->_db, array());
        $db->insert('orders', $data);
        //...
    }
}
//the way it works under the hood:
$provider = new MyProvider();
$db = array($provider, 'getDb');
new Bartender($db);

//inject callback (the PHP 5.3 way)
class Bartender{
    protected $_db;

    public function __construct(Closure $callback){
        $this->_db = $callback;
    }
    public function order($data){
        //validating $data
        //... return or throw exception
        //validation passed, order must be saved
        $db = call_user_func_array($this->_db, array());
        $db->insert('orders', $data);
        //...
    }
}
//the way it works under the hood:
static $conn = null;
$db = function() use ($conn){
    $conn = $conn ? $conn : new mysqli();
    return $conn;
};
new Bartender($db);
Run Code Online (Sandbox Code Playgroud)

Gor*_*onM 7

我最近在计划一个我想尽可能做的主要项目时一直在考虑这个问题(坚持LoD,没有硬编码依赖等).我的第一个想法是"注入工厂"的方法,但我不确定这是要走的路.谷歌的清洁代码谈话声称,如果你通过一个对象来获取你真正想要的对象,那么你就违反了LoD.这似乎排除了注入工厂的想法,因为你必须通过工厂来获得你真正想要的东西.也许我错过了一些让它没问题的观点,但直到我确定我正在考虑其他方法.

你怎么做功能注射?我想你会传入一个回调函数来实现你想要的对象的实例化,但代码示例会很好.

如果您可以使用您提到的三种样式的代码示例来更新您的问题,那么它可能会很有用.我特别渴望看到"注射注射器",即使它是反模式.

  • _到达... _ - 是的; 最重要的是,课程应该"浅薄",并且只引用其他相关的对象; 单体,注射器和工厂不是相关的参考; _你怎么做函数注入?_ - 我有一个特殊的`Callback`类,我作为回调传递(因为PHP中没有内部类); 随着PHP 5.3的出现,现在可以使用lambda函数; 我很快就会得到一些例子 (2认同)
  • @GrigorashVasilij你知道,如果将这个问题转移到programmers.stackexchange.com可能会更好,因为它的术语非常抽象,而stackoverflow实际上应该用于解决更具体的问题(我的实现不起作用,你能不能救命?).你(和我!)可能会得到一些有用的答案. (2认同)

Gor*_*onM 5

确实发生的一个想法是代理对象.它实现了与您想要传入的实际对象相同的接口,但它不是实现任何东西,而是只保存真实类的实例并将方法调用转发给它.

interface MyInterface 
{
    public function doFoo ();
    public function isFoo ();
    // etc
}

class RealClass implements MyInterface
{
    public function doFoo ()
    {
         return ('Foo!');
    }

    public function isFoo ()
    {
        return ($this -> doFoo () == 'Foo!'? true: false);
    }

    // etc
}

class RealClassProxy implements MyInterface
{
    private $instance = NULL;

    /**
     * Do lazy instantiation of the real class
     *
     * @return RealClass
     */
    private function getRealClass ()
    {
        if ($this -> instance === NULL)
        {
            $this -> instance = new RealClass ();
        }
        return $this -> instance;
    }

    public function doFoo ()
    {
        return $this -> getRealClass () -> doFoo ();
    }

    public function isFoo ()
    {
        return $this -> getRealClass () -> isFoo ();
    }

    // etc
}
Run Code Online (Sandbox Code Playgroud)

因为代理具有与真实类相同的接口,所以可以将其作为参数传递给任何为接口键入提示的函数/方法.Liskov替换原则适用于代理,因为它响应所有与真实类相同的消息并返回相同的结果(接口强制执行此操作,至少对于方法签名).但是,除非实际将消息发送到代理,否则实际类不会被实例化,这会在后台对实际类进行惰性实例化.

function sendMessageToRealClass (MyInterface $instance)
{
    $instance -> doFoo ();
}

sendMessageToRealClass (new RealClass ());
sendMessageToRealClass (new RealClassProxy ());
Run Code Online (Sandbox Code Playgroud)

代理对象涉及一个额外的间接层,这显然意味着每个方法调用都会产生很小的性能影响.但是,它确实允许您进行延迟实例化,因此您可以避免实例化您不需要的类.这是否值得,取决于实例化实际对象的成本与额外的间接层的成本.

编辑:我最初编写这个答案的想法是子类化真实对象,所以你可以使用这个技术与没有实现任何接口的对象,如PDO.我原本以为接口是正确的方法,但我想要一种不依赖于类与接口绑定的方法.在反思这是一个很大的错误,所以我更新了答案,以反映我本来应该做的事情.但是,此版本的确意味着您无法直接将此技术应用于没有关联接口的类.您必须将这些类包装在另一个类中,该类确实为代理方法提供了可行的接口,这意味着另一层间接.