- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在处理一个我认为我理解其原因的 DI 问题,但我需要一些建议来解决。
我构建了一个与 Sql 对话的独立程序集(将此程序集称为 a),以及另一个包含业务逻辑的程序集(将此程序集称为 b)。我在 b 程序集中为 db 类创建了一个接口(interface)。由于接口(interface)不是数据库程序集的一部分,我不需要对数据库项目的任何引用,如果我想在运行时运行单元测试而不是程序集,我可以加载对数据库程序集或 stub 的引用需要了解对方。
我可以在编译的业务逻辑库中编写代码,看起来像这样:(假设 a 和 b 是它们各自程序集中的 namespace )
a.IDatabaseClass db_class = (a.IDatabase)new b.Database();
然而,当我尝试运行它时,我得到了一个无效的转换异常。我认为它编译是因为接口(interface)与类完美匹配,但在运行时失败,因为对象签名在数据库类的继承链中看不到 IDatabase。
在 C++ 中,您可以随心所欲地强制转换任何内容,但 C# 在强制转换对象指针方面稍微严格一些。即使该类具有所有正确的函数签名,它也会因为对象不匹配而崩溃。
现在我可以将 db 对象接口(interface)放在带有 db 对象的程序集中,但是业务逻辑需要引用 db 程序集。此外,这只会在未来造成复杂性,因为如果我在单元测试中编写一个 stub 数据库对象,我需要一个对数据库程序集的引用,只是为了我将在我的测试 stub 对象中使用的接口(interface)。这似乎并没有通过这样做来解开耦合......
我可以将所有接口(interface)放在第三个程序集中,该程序集是数据库程序集、业务逻辑和单元测试的父级。这就是解决循环依赖问题的方法。但是,这将 db 程序集与父程序集联系在一起,并使其在与其他项目一起使用时模块化程度大大降低。
我乐于接受有关如何设置每个程序集以使它们独立运行并可用于 DI 的建议。我想我可以将测试 stub 对象保存在与真实代码相同的程序集中,但这看起来很奇怪。
解决方案:下面的一个回复评论说我所追求的基本上是接口(interface)的鸭子类型。 C# 目前不支持鸭子类型,但我认为这可能是可能的,因为接口(interface)实现的行为方式类似于您可能称之为部分类指针(或者更准确地说,函数指针的集合)。我的实验告诉我并非如此,这就是原因。
因此,在 Redmond 将“更多野鸭”放入 c# 之前,看起来我们无法完全达到解耦程序集的最终优雅水平。
最佳答案
创建包含通用接口(interface)的引用库。这样,您将拥有一个包含所有实现不可知逻辑的公共(public)源。
我没有经验可以自信地将此作为绝对陈述,但我强烈怀疑当您谈论引用特定接口(interface)的各个类型及其行为方式时,耦合主要是一个问题。程序集没有那种会使耦合成为问题的特殊性。
让我修改和扩展。一个程序集必须引用另一个程序集,或者两者都必须引用一个公共(public)程序集。 C# 没有鸭子接口(interface),如果那是一件事的话。
现在我认为最好的做法是将业务逻辑与外部接口(interface)隔离开来,因此如果您要执行任何操作,您应该将业务逻辑与数据库实现隔离开来。所以 DB/Application 程序集引用业务逻辑,而不是相反。
关于c# - 跨程序集和命名空间的依赖注入(inject),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34080019/
我已阅读有关依赖注入(inject)的信息。然后来了 构造函数注入(inject), setter/getter 注入(inject) 二传手注入(inject) 接口(interface)注入(in
我正在研究依赖注入(inject)模式。我看过很多例子,其中一个典型的例子是使用 XxxService/XxxRepository 作为例子。但是在我看来,按照UML的概念,类XxxRepositor
我开始使用 Google Guice。 我有一个简单的问题: javax.inject 的 @Inject 注释和 com.google.inject 的 有什么区别@Inject 一个 ? 谢谢。
当使用构造函数注入(inject)工厂方法时,依赖的属性不会得到解析。但是,如果在解析依赖的组件之前解析了工厂方法,则一切都会按预期工作。此外,当仅使用属性注入(inject)或构造函数注入(inje
我有这样的事情: class Root { public Root(IDependency dep) {} } class Dependency:IDependency { p
听完Clean Code Talks ,我开始明白我们应该使用工厂来组合对象。因此,例如,如果 House有一个 Door和 Door有一个 DoorKnob , 在 HouseFactory我们创建
情况:我需要在一些 FooClass 中进行惰性依赖实例化,所以我通过 Injector类作为构造函数参数。 private final Injector m_injector; public Foo
在编写代码时,我们应该能够识别两大类对象: 注入(inject)剂 新品 http://www.loosecouplings.com/2011/01/how-to-write-testable-cod
这个问题是关于 Unity Container 的,但我想它适用于任何依赖容器。 我有两个具有循环依赖关系的类: class FirstClass { [Dependency] pub
如果我有 10 个依赖项我需要注入(inject)并且不想在构造函数中有 10 个参数,我应该使用哪种注入(inject)模式? public class SomeClass { privat
我在使用 Angular2 DI 时遇到了问题。我尝试将一个类注入(inject)另一个类,它引发了以下错误: 留言:"Cannot resolve all parameters for 'Produ
对依赖注入(inject)还很陌生,我想弄清楚这是否是一种反模式。 假设我有 3 个程序集: Foo.Shared - this has all the interfaces Foo.Users -
我正在尝试了解 Angular 14 的变化,尤其是 inject()我可以将模块注入(inject)功能的功能,我不需要为此创建特殊服务..但我想我弄错了。 我正在尝试创建一些静态函数来使用包 ng
希望这个问题不是太愚蠢,我试图掌握更高级的编程原理,因此试图习惯使用 Ninject 进行依赖注入(inject)。 因此,我的模型分为几个不同的 .dll 项目。一个项目定义了模型规范(接口(int
我最近一直在大量使用依赖注入(inject)、测试驱动开发和单元测试,并且开始喜欢上它。 我在类中使用构造函数依赖,这样我就可以为单元测试注入(inject)模拟依赖。 但是,当您实际需要生产环境中的
我有下面的代码来使用 Guice 进行依赖注入(inject)。第一个是使用构造函数注入(inject),而另一个是直接在字段上方添加 @Inject。这两种方式有什么区别吗? Guice官网似乎推荐
这个问题在这里已经有了答案: Angular2 Beta dependency injection (3 个答案) 关闭 7 年前。 我正在使用 angular2 测试版。并在使用 @Inject
有没有可能做这样的事情? (因为我尝试过,但没有成功): @Injectable() class A { constructor(private http: Http){ // <-- Injec
我很恼火必须通过 Constructor 传递管道对象,因为我想为业务实体或要传递的值保留构造函数参数。 所以我想通过 setter ,但只要这些 setter 没有被填充,我的包含依赖项的对象就不应
假设我有这个: SomePage.razor: @inject Something something @page "/somepage" My Page @code { // Using
我是一名优秀的程序员,十分优秀!