- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用C#中的小型(ish)支持库来开发一个项目(可能还有类似的项目)。我已经阅读了很多有关构造函数注入依赖项的知识,虽然我理解了其背后的基本动机,但在处理某些类型的依赖项时这是很有意义的,但我经常遇到这样的情况:它似乎使事情变得更加困难而没有提供很多好处。
我想展示两个示例,非常感谢您提供有关如何“正确”处理它们的任何建议。我还要在这一点上指出,目前我不希望为此使用任何DI容器,因为我希望该库在没有容器的情况下可以使用...
示例1:动态创建的对象
大多数与DI相关的文章都强调,对象图的创建发生在合成的根部。但是,并非所有对象都可以最初创建。对此的标准答案是将工厂用于短期对象。当然可以,但是在某些情况下可能会感到很尴尬...
假设我想在第三方库中的对象周围创建一系列外观类,其中一个对象可能返回另一种类型的对象,并且我想将两者都包装起来。作为一个(组成的)示例,让我们采用一个具有方法“ GetFiles()”的“ Directory”类,该方法返回“ File”类的实例。外墙的非DI版本可能如下所示:
public DirectoryFacade(string path)
{
m_directory = new Directory(path);
}
public IEnumerable<FileFacade> GetFiles()
{
foreach(File file in m_directory.GetFiles())
{
yield return new FileFacade(file);
}
}
Configuration config = new Configuration(configString);
config.SetValueForKey('foo','bar');
...
KeywordFactory factory = new KeywordFactory();
ConfigurationParser parser = new ConfigurationParser(factory);
Configuration config = new Configuration(parser);
config.ConfigurationString = configString; //NOTE: this could be a constructor param...
config.SetValueForKey('foo','bar');
最佳答案
现在使用DI,DirectoryFacade本身不应创建任何FileFacades
这是您出问题的地方。防止在应用程序中的任何位置使用new
关键字绝不是DI的意图。 DI是关于松散耦合的。而且,您不应该尝试抽象那些不必抽象的事物。您应该区分服务(应用程序逻辑)和其他任何东西(DTO,消息,实体,命令,异常,原语等)。这是newables and injectables之间的重要区别。
在这种情况下,您的FileFacades
是可更新的。它们是短命的,仅包含数据,没有自身的依赖关系,几乎没有逻辑可以测试,并且主要作为实际实现中的瘦代理存在。注入它们意味着您对将它们替换为其他实现(例如用于测试)感兴趣。但是,如果不是这种情况,请不要打扰他们。在这种情况下,让DirectoryFacade
创建FileFacade
实例似乎是完全合理的。这完全消除了使用FileFacadeFactory
的需要。
因此,关键字可能是由“ KeywordFactory”生成的...
这种说法再次来自同样的误解。关键字可能只是数据包。这里没有抽象的逻辑,也没有理由解释为什么ConfigurationParser
不应重新创建Keyword
实例本身。
可以再写一些其他的工厂和门面来
提供客户端的默认实现,但是这样做
真的有意义吗?它提供什么好处?
从另一侧看。如果您不像这样构造代码,那么如何对代码进行单元测试?
正如我所看到的,作为框架编写者,您有两个责任。 1.编写正确运行的代码(通过对它进行彻底的单元测试)。 2.设计一个易于客户使用的API。
如果所有这些类都是实现细节,并且不打算让客户端更改,请确保它们保留实现细节。例如,将这些类设置为内部类,并对库的根类型使用穷人的依赖注入。对于穷人的DI,您有一个公共默认构造函数,该构造函数调用另一个接受所有依赖关系的构造函数。这是一个例子:
public class MyPublicAPI
{
public MyPublicAPI()
: this(new Configuration(
new ConfigurationParser()))
{
}
internal MyPublicAPI(IConfiguration configuration)
{
this.configuration = configuration;
}
}
关于c# - 努力进行手动依赖注入(inject),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18309265/
我正在尝试为我的项目创建一些单元测试,经过大量挖掘之后,我发现了Effort,这个想法很棒,它模拟数据库而不是处理伪造的DBContext,顺便说一句,很难做到正确使用复杂的架构。 但是,我将用户的电
我正在尝试对某些响应 Entity Framework 数据库上下文的类进行单元测试。为了寻求帮助,我设法找到了一个名为 Effort 的库,它似乎有点旧,而且没有很好的文档记录,但它似乎可以工作,而
这两天,Auto-GPT 爆火 https://github.com/Torantulino/Auto-GPT 它是一款让最强语言模型GPT-4能够自主完成任务的模型,让整个AI圈疯
为什么会出现这个异常?这是错误吗? 我正在使用 EF 测试库 Effort 创建我的数据库的内存实例并遇到这个有趣的场景: 打开DbContext1 将项目添加到表(不保存) 关闭DbContext1
我是一名优秀的程序员,十分优秀!