如何避免在UI层中引用DataLayer?

问题描述

| 我有一个3层的项目。 UI,业务层和数据层。 UI调用BL。 BL呼叫DL。 DL执行数据库操作。就这么简单。 我想对BL方法进行单元测试,因此我做了一些更改,现在我在BL构造函数中接受DL作为参数,以便可以创建DL的Mock对象。 这使我在UI调用BL时更改了UI层,并且根据体系结构规则,如果我将DL的引用添加到UI中,我认为它不是一个好的设计。 有人可以提出更好的方法吗?我需要更改体系结构还是在这里做错了什么?我可以在这里介绍Facade Manager吗?您的建议的一个例子将不胜感激。 - 编辑 - 这是代码: 在BL中:
    public MyBusinessLayer()
    {

    }
    //To pass mock object of WCF Service
    public MyBusinessLayer(ISomeServices svc)
    {
        someServiceRef = svc;
    }

    //To pass mock object of Data Layer.
    public MyBusinessLayer(ISomeDataLayer dl)
    {
        someDlRef = dl;
    }
在用户界面中:
//To do this i have to add DL reference to UI
MyBusinessLater b = new MyBusinessLayer(new ISomeService());
    

解决方法

IoC非常适合这项工作。您的BL应该仅依靠可以获取所需数据的接口。然后,使用IoC向接口注册实际的数据库实现。然后,应该为UI注入接口BL。因此,在应用程序启动时,您注册了所有依赖项,并将BL接口注入到UI中,因此UI不知道实际的BL实现依赖于DL接口这一事实。 这只是我在网上找到的示例。只是谷歌的C#IOC或C#依赖注入。 http://www.dotnetspark.com/kb/266-inversion-control-ioc-and-dependency-injection.aspx     ,一种方法是将所有数据类型放在单独的dll中。 然后从UI,BL和DAL引用此dll。 这样就不需要UI引用DAL。     ,@BFree的答案是正确的;我只是在回答以澄清我的评论。这是我的建议(我认为与@BFree的建议相同):
public class MyBusinessLayer : IBusinessLayer
{
    public MyBusinessLayer(ISomeDataLayer dl,ISomeServices svc) {}
}

public class MyUI
{
    public MyUI(IBusinessLayer bl) {}
}
无论您使用的是MVC,MVVM还是其他任何东西,某些地方都在创建您的用户界面(或其控制器)。该方法可以了解所有内容,因为它可以将所有内容连接起来。其他所有信息都只涉及接口。 与项目中的程序集引用相比,我更关心在其他层(耦合)中对具体类型的引用。     

相关问答

依赖报错 idea导入项目后依赖报错,解决方案:https://blog....
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下...
错误1:gradle项目控制台输出为乱码 # 解决方案:https://bl...
错误还原:在查询的过程中,传入的workType为0时,该条件不起...
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct...