依赖注入 – Mark Seemann关于Bastard Injection的矛盾陈述 需要一些澄清

我正在读他的书 Dependency Injection in Net

1)Here他说,只有当我们使用Foreign Default时,才会发生混蛋注射。

但是在他的书中,第148页的图示显示,当依赖关系的认实现是Foreign Default或Local Default时,会发生Bastard注入:

那么当认执行依赖关系是一个本地认值时,Bastard Injection反模式也会发生吗?

2)Here(也在他的书中)他指出,一个类有一个可选的依赖关系,只要这个依赖关系的认实现是一个很好的局部认:

但在下一个article,他似乎反对有可选的依赖关系,即使认实现是一个本地认:

06000

In terms of encapsulation,the main problem with such an approach is
that it seems like the MyConsumer class can’t really make up its mind
whether or not it controls the creation of its log dependency. While
this is a simplified example,this Could become a problem if the ILog
instance returned by LogManager wraps an unmanaged resource which
should be disposed when it’s no longer needed.

认执行依赖关系是本地的时候,他上述摘录中的论证也是有效的吗?如果是这样,那么还应避免使用本地缺省值的可选依赖关系?

3)
PG。 147:

The main problem with Bastard Injection is its use of a FOREIGN
DEFAULT …,we can no longer freely reuse the class because it drags
along a dependency we may not want. It also becomes more difficult to
do parallel development because the class depends strongly on its
DEPENDENCY.

外部认是一个依赖关系的实现,它用作认值,并且在不同于其消费者的程序集中定义。因此,在外来违约方面,消费者集会也将拖累依赖关系的集会。

他是否也意味着“外部违约”使并行开发更加困难,而“本地认”不是?如果他是这样,那么这是没有意义的,因为我会假设并行开发困难的原因并不在于消费者的大会很难提及依赖组织,而是消费类依赖于具体实现的事实依赖?

谢谢

既然在这里有很多问题,我将首先尝试提供一个关于我的观点的综合,然后根据这个材料明确地回答每个问题。

合成

当我写了the book,我首先试图描述我在野外目睹的模式和反模式。因此,书中的模式和反模式首先是描述性的,只有较小程度的规定性。显然,将它们分为模式和反模式意味着一定程度的判断力:)

Bastard Injection有多个层面的问题:

>软件包依赖关系
>封装
>使用方便

最危险的问题与包依赖关系有关。这是我通过引入“外部认值”和“本地认值”这个术语来尝试更有效的概念。外部认值的问题是它们拖动硬耦合的依赖关系,其中makes (de/re)composition impossible.一个很好的资源,更明确地处理包管理是Agile Principles,Patterns,and Practices

在封装级别上,这样的代码很难理解:

private readonly ILog log;
public MyConsumer(ILog log)
{
    this.log = log ??LogManager.GetLogger("My");
}

虽然它保护类的不变量,但问题是在这种情况下,null是可接受的输入值。情况并非如此。在上面的例子中,LogManager.GetLogger(“My”)可能只会引入本地缺省值。从这个代码片段,我们无法知道这是否是真的,但是为了参数,让我们假设现在。如果认的ILog确实是Local Default,则MyConsumer的客户端可以传入null而不是ILog。请记住,封装是关于使客户端轻松使用对象而不了解所有实现细节。这意味着这是客户端所看到的:

public MyConsumer(ILog log)

在C#(和类似的语言)中,可以传递null而不是ILog,它将要编译:

var mc = new MyConsumer(null);

通过上述实现,不仅将编译,而且在运行时也起作用。据Postel’s law,这是件好事,对吧?

不幸的是,它不是。

考虑另一个具有所需依赖关系的类;我们把它称为Repository,只是因为这是一个众所周知的(虽然过度使用)的模式:

private readonly IRepository repository;
public MyOtherConsumer(IRepository repository)
{
    if (repository == null)
        throw new ArgumentNullException("repository");

    this.repository = repository;
}

与封装保持一致,客户端只能看到:

public MyOtherConsumer(IRepository repository)

根据以往的经验,程序员可能会倾向于编写如下代码

var moc = new MyOtherConsumer(null);

这仍然编译,但在运行时失败!

你如何区分这两个构造函数

public MyConsumer(ILog log)
public MyOtherConsumer(IRepository repository)

您不能,但目前,您有不一致的行为:在一种情况下,null是一个有效的参数,但在另一种情况下,null将导致运行时异常。这将减少每个客户端程序员在API中具有的信任。一贯是更好的前进方向。

为了使类似MyConsumer的类更容易使用,您必须保持一致。这就是为什么接受null是一个坏主意。一个更好的方法是使用构造函数链接

private readonly ILog log;

public MyConsumer() : this(LogManager.GetLogger("My")) {}

public MyConsumer(ILog log)
{
    if (log == null)
        throw new ArgumentNullException("log");

    this.log = log;
}

客户现在看到:

public MyConsumer()
public MyConsumer(ILog log)

这与MyOtherConsumer是一致的,因为如果您尝试传递null而不是ILog,您将收到运行时错误

虽然这在技术上仍然是Bastard Injection,I can live with this design用于本地认值;事实上,我有时会设计这样的API,因为它是一种众所周知的许多语言成语。

出于许多目的,这是足够好的,但仍然违反了重要的设计原则:

Explicit is better than implicit

虽然构造函数链可以让客户端使用认的ILog使用MyConsumer,但是没有一个简单的方法来弄清楚ILog的认实例是什么。有时,这也很重要。

此外,认构造函数的存在会暴露出一段代码将在Composition Root之外调用认构造函数的风险。如果发生这种情况,您已经过早地将对象耦合到对方,一旦你完成了,您不能将它们从组合根中分离。

因此,使用简单构造函数注入的风险较小:

private readonly ILog log;

public MyConsumer(ILog log)
{
    if (log == null)
        throw new ArgumentNullException("log");

    this.log = log;
}

您仍然可以使用认记录器编写MyConsumer:

var mc = new MyConsumer(LogManager.GetLogger("My"));

如果要使本地认更容易被发现,您可以将其作为工厂在某处,例如在MyConsumer类本身上:

public static ILog CreateDefaultLog()
{
    return LogManager.GetLogger("My");
}

所有这一切都为回答这个问题中的具体子问题奠定了基础。

1.当认执行依赖关系是本地认值时,是否还会发生混蛋注入反模式?

是的,在技术上,它确实,但后果不那么严重。混蛋注射首先是一个描述,使您能够轻松识别它,当你遇到它。

请注意,本书上面的图示描述了如何从Bastard Injection重构;不是如何识别它。

2. [应该]还可以避免使用本地认值的可选依赖关系[…]

从包依赖的角度来看,你不需要避免这些;他们比较温和

从使用的角度来看,我仍然倾向于避开他们,但这取决于我正在建设什么。

>如果我创建了许多人可以使用的可重用库(例如OSS项目),我仍然可以选择构造函数链接,以便更容易地开始使用API​​。
>如果我正在创建一个仅在特定代码库中使用的类,我倾向于完全避免可选的依赖关系,而是构成组合根目录中的所有内容

他是否也意味着“外部违约”使并行开发更加困难,而“本地认”不是?

不,我不如果您有认值,则认值必须在使用之前就位;无论是本地还是外国都无关紧要。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


迭代器模式(Iterator)迭代器模式(Iterator)[Cursor]意图:提供一种方法顺序访问一个聚合对象中的每个元素,而又不想暴露该对象的内部表示。应用:STL标准库迭代器实现、Java集合类型迭代器等模式结构:心得:迭代器模式的目的是在不获知集合对象内部细节的同时能对集合元素进行遍历操作
高性能IO模型浅析服务器端编程经常需要构造高性能的IO模型,常见的IO模型有四种:(1)同步阻塞IO(BlockingIO):即传统的IO模型。(2)同步非阻塞IO(Non-blockingIO):默认创建的socket都是阻塞的,非阻塞IO要求socket被设置为NONBLOCK。注意这里所说的N
策略模式(Strategy)策略模式(Strategy)[Policy]意图:定义一系列算法,把他们封装起来,并且使他们可以相互替换,使算法可以独立于使用它的客户而变化。应用:排序的比较方法、封装针对类的不同的算法、消除条件判断、寄存器分配算法等。模式结构:心得:对对象(Context)的处理操作可
访问者模式(Visitor)访问者模式(Visitor)意图:表示一个作用于某对象结构中的各元素的操作,它使你在不改变各元素的类的前提下定义作用于这些元素的新操作。应用:作用于编译器语法树的语义分析算法。模式结构:心得:访问者模式是要解决对对象添加新的操作和功能时候,如何尽可能不修改对象的类的一种方
命令模式(Command)命令模式(Command)[Action/Transaction]意图:将一个请求封装为一个对象,从而可用不同的请求对客户参数化。对请求排队或记录请求日志,以及支持可撤消的操作。应用:用户操作日志、撤销恢复操作。模式结构:心得:命令对象的抽象接口(Command)提供的两个
生成器模式(Builder)生成器模式(Builder)意图:将一个对象的构建和它的表示分离,使得同样的构建过程可以创建不同的表示。 应用:编译器词法分析器指导生成抽象语法树、构造迷宫等。模式结构:心得:和工厂模式不同的是,Builder模式需要详细的指导产品的生产。指导者(Director)使用C
设计模式学习心得《设计模式:可复用面向对象软件的基础》一书以更贴近读者思维的角度描述了GOF的23个设计模式。按照书中介绍的每个设计模式的内容,结合网上搜集的资料,我将对设计模式的学习心得总结出来。网络上关于设计模式的资料和文章汗牛充栋,有些文章对设计模式介绍生动形象。但是我相信“一千个读者,一千个
工厂方法模式(Factory Method)工厂方法模式(Factory Method)[Virtual Constructor]意图:定义一个用于创建对象的接口,让子类决定实例化哪一个类,使一个类的实力化延迟到子类。应用:多文档应用管理不同类型的文档。模式结构:心得:面对同一继承体系(Produc
单例模式(Singleton)单例模式(Singleton)意图:保证一个类只有一个实例,并提供一个访问它的全局访问点。应用:Session或者控件的唯一示例等。模式结构:心得:单例模式应该是设计模式中最简单的结构了,它的目的很简单,就是保证自身的实例只有一份。实现这种目的的方式有很多,在Java中
装饰者模式(Decorator)装饰者模式(Decorator)[Wrapper]意图:动态的给一个对象添加一些额外的职责,就增加功能来说,比生成子类更为灵活。应用:给GUI组件添加功能等。模式结构:心得:装饰器(Decorator)和被装饰的对象(ConcreteComponent)拥有统一的接口
抽象工厂模式(Abstract Factory)抽象工厂模式(Abstract Factory)[Kit]意图:提供一个创建一系列相关或相互依赖对象的接口,而无须指定他们具体的类。应用:用户界面工具包。模式结构:心得:工厂方法把生产产品的方式封装起来了,但是一个工厂只能生产一类对象,当一个工厂需要生
桥接模式(Bridge)桥接模式(Bridge)[Handle/Body]意图:将抽象部分与它的实现部分分离,使他们都可以独立的变化。应用:不同系统平台的Windows界面。模式结构:心得:用户所见类体系结构(Window派生)提供了一系列用户的高层操作的接口,但是这些接口的实现是基于具体的底层实现
适配器模式(Adapter)适配器模式(Adapter)[Wrapper]意图:将类的一个接口转换成用户希望的另一个接口,使得原本由于接口不兼容而不能一起工作的类可以一起工作。应用:将图形类接口适配到用户界面组件类中。模式结构:心得:适配器模式一般应用在具有相似接口可复用的条件下。目标接口(Targ
组合模式(Composition)组合模式(Composition)意图:将对象组合成树形结构以表示“部分-整体”的层次结构,使得用户对单个对象和组合对象的使用具有一致性。应用:组合图形、文件目录、GUI容器等。模式结构:心得: 用户(Client)通过抽象类(Component)提供的公用接口统一
原型模式(Prototype)原型模式(Prototype)意图:用原型实例制定创建对象的种类,并且通过拷贝这些原型创建新的对象。应用:Java/C#中的Clonable和IClonable接口等。模式结构:心得:原型模式本质上就是对象的拷贝,使用对象拷贝代替对象创建的原因有很多。比如对象的初始化构
什么是设计模式一套被反复使用、多数人知晓的、经过分类编目的、代码 设计经验 的总结;使用设计模式是为了 可重用 代码、让代码 更容易 被他人理解、保证代码 可靠性;设计模式使代码编制  真正工程化;设计模式使软件工程的 基石脉络, 如同大厦的结构一样;并不直接用来完成代码的编写,而是 描述 在各种不同情况下,要怎么解决问题的一种方案;能使不稳定依赖于相对稳定、具体依赖于相对抽象,避免引
单一职责原则定义(Single Responsibility Principle,SRP)一个对象应该只包含 单一的职责,并且该职责被完整地封装在一个类中。Every  Object should have  a single responsibility, and that responsibility should be entirely encapsulated by t
动态代理和CGLib代理分不清吗,看看这篇文章,写的非常好,强烈推荐。原文截图*************************************************************************************************************************原文文本************
适配器模式将一个类的接口转换成客户期望的另一个接口,使得原本接口不兼容的类可以相互合作。
策略模式定义了一系列算法族,并封装在类中,它们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。