- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我目前有一些空闲时间,所以我正在努力了解 DI 和 IoC 容器。我选择 unity 没有任何理由,除了我可以告诉我的主要框架之间没有重大差异,我应该太担心开始。随着事情变得越来越复杂,我意识到我可能需要做出改变,但现在我希望它能做到。
所以,我正在处理一个相对简单的数据访问场景,并实现了以下接口(interface)和数据访问类。
public interface IEventRepository
{
IEnumerable<Event> GetAll();
}
public class EventRepository : IEventRepository
{
public IEnumerable<Event> GetAll()
{
// Data access code here
}
}
然后使用我可以执行以下操作。
IUnityContainer container = new UnityContainer();
container.RegisterType(typeof(IEventRepository), typeof(EventRepository));
var eventRepo = container.Resolve<IEventRepository>();
eventRepo.GetAll();
如果我需要在 6 个月内根据我的理解更改我的数据库提供程序,我会创建一个新的 IEventRepository 实现并更新类型注册,这很好。
现在,这就是我感到困惑的地方。例如,如果我想实现一些缓存,我可以从 IEventRepository 的适当实现继承并覆盖适当的方法来实现必要的缓存。但是,这样做会使使用通过 DI 传入的 Moq 实现测试缓存是否正常工作变得更加困难,因此本着 DI 的真正精神,我认为创建 IEventRepository 的实现,然后使用 DI 请求一个像这样的 IEventRepository 的实际数据访问实现。
public class CachedEventRepository : IEventRepository
{
private readonly IEventRepository _eventRepo;
public CachedEventRepository(IEventRepository eventRepo)
{
if (eventRepo is CachedEventRepository)
throw new ArgumentException("Cannot pass a CachedEventRepository to a CachedEventRepository");
_eventRepo = eventRepo;
}
public IEnumerable<Event> GetAll()
{
// Appropriate caching code ultimately calling _eventRepo.GetAll() if needed
}
}
这是有道理的还是我做错了?你有什么建议?如果我做得正确,我该如何解决以下情况,以便 CachedEventRepository 获得 IEventRepository 的适当数据访问实现?
IUnityContainer container = new UnityContainer();
container.RegisterType(typeof(IEventRepository), typeof(EventRepository));
container.RegisterType(typeof(IEventRepository), typeof(CachedEventRepository));
var eventRepo = container.Resolve<IEventRepository>();
eventRepo.GetAll();
非常感谢您的帮助。
编辑 1以下是我希望能够执行的最小起订量测试,我认为使用继承不可能实现,并且需要 DI。
var cacheProvider = new MemoryCaching();
var eventRepo = new Mock<IEventRepository>(MockBehavior.Strict);
eventRepo
.Setup(x => x.GetAll())
.Returns(() =>
{
return new Event[] {
new Event() { Id = 1},
new Event() { Id = 2}
};
});
var cachedEventRepo = new CachedEventRepository(
eventRepo.Object,
cacheProvider);
var data = cachedEventRepo.GetAll();
data = cachedEventRepo.GetAll();
data = cachedEventRepo.GetAll();
Assert.IsTrue(data.Count() > 0);
eventRepo.Verify(x => x.GetAll(), Times.Once());
// This set method should expire the cache so next time get all is requested it should
// load from the database again
cachedEventRepo.SomeSetMethod();
data = cachedEventRepo.GetAll();
data = cachedEventRepo.GetAll();
Assert.IsTrue(data.Count() > 0);
eventRepo.Verify(x => x.GetAll(), Times.Exactly(2));
最佳答案
好的,在对这个主题进行了一些思考并研究了 Unity 之后,我想到了这个。
public class EventRepository : IEventRepository
{
private readonly IDbManager _dbManager;
public EventRepository(IDbManager dbManager)
{
_dbManager = dbManager;
}
public virtual IEnumerable<Event> GetAll()
{
// Data access code
}
}
public class CachedEventRepository : IEventRepository
{
private readonly ICacheProvider _cacheProvider;
private readonly IEventRepository _eventRepo;
public ICacheProvider CacheProvider
{
get { return _cacheProvider; }
}
public CachedEventRepository(IEventRepository eventRepo, ICacheProvider cacheProvider)
{
if(eventRepo is CachedEventRepository)
throw new ArgumentException("eventRepo cannot be of type CachedEventRepository", "eventRepo");
_cacheProvider = cacheProvider;
_eventRepo = eventRepo;
}
public IEnumerable<Event> GetAll()
{
// Caching logic for this method with a call to _eventRepo.GetAll() if required
}
}
这需要下面的unity注册。对 IEventRepository 的解析请求将返回一个 CachedEventRepository。如果我想快速删除缓存,我只需删除 CachedEventRepository 注册,它将恢复为 EventRepository。
IUnityContainer container = new UnityContainer();
container.RegisterType<IDbManager, SqlDbManager>();
container.RegisterType<ICacheProvider, MemoryCaching>();
container.RegisterType<IEventRepository, EventRepository>();
container.RegisterType<IEventRepository, CachedEventRepository>(
new InjectionConstructor(
new ResolvedParameter<EventRepository>(),
new ResolvedParameter<ICacheProvider>())
);
然后这将允许我进行的测试。
一个简单的数据访问测试...SQL 是否工作
IUnityContainer container = new UnityContainer();
container.RegisterType<IDbManager, SqlDbManager>();
container.RegisterType<EventRepository>();
var repo = container.Resolve<EventRepository>();
var data = repo.GetAll();
Assert.IsTrue(data.Count() > 0);
一个简单的缓存测试...缓存系统是否工作
var cache = new MemoryCaching();
var getVal = cache.Get<Int32>(
"TestKey",
() => { return 2; },
DateTime.UtcNow.AddMinutes(5));
Assert.AreEqual(2, getVal);
getVal = cache.Get<Int32>(
"TestKey",
() => { throw new Exception("This should not be called as the value should be cached"); },
DateTime.UtcNow.AddMinutes(5));
Assert.AreEqual(2, getVal);
以及对两者协同工作的测试......在各个方法上的缓存是否按预期工作。缓存是否在应该过期的时候过期,方法参数是否正确工作以触发新的数据库请求等。
var cacheProvider = new MemoryCaching();
var eventRepo = new Mock<IEventRepository>(MockBehavior.Strict);
eventRepo
.Setup(x => x.GetAll())
.Returns(() =>
{
return new Event[] {
new Event() { Id = 1},
new Event() { Id = 2}
};
});
var cachedEventRepo = new CachedEventRepository(
eventRepo.Object,
cacheProvider);
cachedEventRepo.CacheProvider.Clear();
var data = cachedEventRepo.GetAll();
data = cachedEventRepo.GetAll();
data = cachedEventRepo.GetAll();
Assert.IsTrue(data.Count() > 0);
eventRepo.Verify(x => x.GetAll(), Times.Once());
cachedEventRepo.SomeSetMethodWhichExpiresTheCache();
data = cachedEventRepo.GetAll();
data = cachedEventRepo.GetAll();
Assert.IsTrue(data.Count() > 0);
eventRepo.Verify(x => x.GetAll(), Times.Exactly(2));
你怎么看这个?我认为它提供了良好的分离和良好的可测试性。
关于c# - 对 DI、IoC、Unity 和 Moq 的类型注册和实现感到困惑,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10231537/
我有一个问题,想知道如何在已经使用常规登录/注册系统的网站上集成第 3 方登录(也进行静默注册)。 基本上当前登录还是比较规律的: 当用户进入网站 session 时类(class)决定他是否需要重新
我在学习 Hyperledger Fabric 并运行示例代码。 我仍在尝试正确了解事情的运作方式,尤其是在使用证书和加密 Material 的用户/管理员注册和注册中。 我想知道以下如何工作。 1)
我正在尝试使用从我的 Android 设备的 PassWallet 应用程序中保存的票证中获取的 token 向我的设备发送消息。 设备发送使用苹果钱包规范更新通行证所需的所有信息。但是,我正在使用
使用 Passport 本地示例,我可以登录工作。没有关于如何注册用户的文档。 我想为用户提供一个“电子邮件”和“密码”字段,他们可以使用它们来注册该网站。我怎样才能做到这一点?有什么原因没有记录下来
在之前的一些 WSO2IS 版本中,有一个默认的 self 注册功能。但是,我在 5.0 版本中找不到它。 阅读 WSO2IS 5.0 文档,我发现有 2 个用于此功能的 API: getUserId
我已将 Airship SDK 集成到 Android 应用程序中。在应用程序启动和飞艇起飞后,我在日志中得到以下调试信息: 07-27 12:46:31.916 XXX - UALib( 1545)
Delphi 中设计时包的可怕错误之一是以下错误,这意味着注册安装新组件到您的 Palette 上的包: Component TSomething can't be registered by pac
我发现的大部分内容都使用 php 或类似的东西。 我有一个 Angular 前端和 Node/express 服务器代码。还没有后端。我不确定如何继续用户注册。 最佳答案 在没有后端的情况下进行注册等
我正在使用 Drupal 6 开发一个网站。我正在使用我自己的主题,并且一切正常。现在我需要使用我的自定义主题在我的网站上创建一个自定义登录/注册表单。我尝试了很多方法,但一切都重定向到我的管理主题,
我在运行此代码时收到 Sip 异常。这是因为 manager.register(me,20,listener)。 下面是我的代码,所以请帮我更正这段代码。我正在使用 SipDemo 代码注册我的帐户。
在过去的几天里,我一直在尝试为基于 VUE 的 excel 制作任务 Pane 插件。 我已按照 link 的指南进行操作我试图为 onSelectionChange 注册一个事件处理程序。它已经有些
我需要在使用 django-registration 应用程序的登录表单中实现一个“记住我”按钮。任何ane可以帮助我向我展示这样做的方法吗? 谢谢 最佳答案 一种方法是更改 session 到期
我发现,如果您使用 Django 1.5 版本,则 django 注册模块会中断,因为在最新的 django 开发版本中,simple.py 类已被删除。 最佳答案 此问题现已修复: hg clone
我正在尝试将 facebook connect 实现到我的网站,但有几个问题。 1:是否可以使用用户当前的 Facebook 电子邮件/密码在我的网站上注册用户。 假设用户点击链接通过 faceboo
我使用 Facebook 注册来允许人们在我的网站上注册。有没有可能,在注册后,他在我的网站注册的成员(member)墙上会张贴? 最佳答案 这可能不是您正在寻找的答案,但我强烈建议您不要这样做。用户
I would like to use a slash (/) for a search during a vimscript, but I don't want to overwrite the "
我正在使用 jqgrid 并且有显示日期的列,但是来自服务器的日期以 json 格式出现,如下所示, "CommentedDate": "\/Date(1304324941000+0530)\/" 如
我希望用户可以直接登录主页,而不是在“../account/login/”页面上登录。我应该做什么才能使它成为可能?如何将主页上的输入字段与 allauth 连接?我不知道这样是否太复杂而无法以这种方
This question already has answers here: Understanding NSString comparison (7个答案) 5年前关闭。 我正在尝试制作注册表。有
我正在使用 Django 注册。它提供了处理 registration_form.html 的 View ,该 html 当前包含用户名、密码 1、密码 2 和电子邮件作为我的应用程序中的用户可输入字
我是一名优秀的程序员,十分优秀!