【发布时间】:2015-06-30 10:18:09
【问题描述】:
我正在尝试在我的 Windows 窗体应用程序中实现 IoC。我的选择落在了 Simple Injector 上,因为它又快又轻。我还在我的应用程序中实现了工作单元和存储库模式。这是结构:
DbContext:
public class MemberContext : DbContext
{
public MemberContext()
: base("Name=MemberContext")
{ }
public DbSet<Member> Members { get; set; }
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Conventions.Remove<PluralizingTableNameConvention>();\
}
}
型号:
public class Member
{
public int MemberID { get; set; }
public string Name { get; set; }
}
GenericRepository:
public abstract class GenericRepository<TEntity> : IGenericRepository<TEntity>
where TEntity : class
{
internal DbContext context;
internal DbSet<TEntity> dbSet;
public GenericRepository(DbContext context)
{
this.context = context;
this.dbSet = context.Set<TEntity>();
}
public virtual void Insert(TEntity entity)
{
dbSet.Add(entity);
}
}
会员存储库:
public class MemberRepository : GenericRepository<Member>, IMemberRepository
{
public MemberRepository(DbContext context)
: base(context)
{ }
}
工作单位:
public class UnitOfWork : IUnitOfWork
{
public DbContext context;
public UnitOfWork(DbContext context)
{
this.context = context;
}
public void SaveChanges()
{
context.SaveChanges();
}
private bool disposed = false;
protected virtual void Dispose(bool disposing)
{
if (!this.disposed)
{
if (disposing)
{
context.Dispose();
}
}
this.disposed = true;
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
}
会员服务:
public class MemberService : IMemberService
{
private readonly IUnitOfWork unitOfWork;
private readonly IMemberRepository memberRepository;
public MemberService(IUnitOfWork unitOfWork, IMemberRepository memberRepository)
{
this.unitOfWork = unitOfWork;
this.memberRepository = memberRepository;
}
public void Save(Member member)
{
Save(new List<Member> { member });
}
public void Save(List<Member> members)
{
members.ForEach(m =>
{
if (m.MemberID == default(int))
{
memberRepository.Insert(m);
}
});
unitOfWork.SaveChanges();
}
}
在成员表单中,我只添加了一个文本框来输入成员名称和一个按钮来保存到数据库。这是会员形式的代码:
frmMember:
public partial class frmMember : Form
{
private readonly IMemberService memberService;
public frmMember(IMemberService memberService)
{
InitializeComponent();
this.memberService = memberService;
}
private void btnSave_Click(object sender, EventArgs e)
{
Member member = new Member();
member.Name = txtName.Text;
memberService.Save(member);
}
}
我在 Program.cs 中实现了 SimpleInjector(参考 http://simpleinjector.readthedocs.org/en/latest/windowsformsintegration.html),如下面的代码所示:
static class Program
{
private static Container container;
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Bootstrap();
Application.Run(new frmMember((MemberService)container.GetInstance(typeof(IMemberService))));
}
private static void Bootstrap()
{
container = new Container();
container.RegisterSingle<IMemberRepository, MemberRepository>();
container.Register<IMemberService, MemberService>();
container.Register<DbContext, MemberContext>();
container.Register<IUnitOfWork, UnitOfWork>();
container.Verify();
}
}
当我运行程序并添加成员时,它不会保存到数据库中。如果我将container.Register 更改为container.RegisterSingle,它将保存到数据库。从文档中,RegisterSingle 将使我的班级成为单身人士。我不能使用 RegisterLifeTimeScope 因为它会产生错误
“IMemberService 类型的注册委托引发了异常。IUnitOfWork 注册为“Lifetime Scope”生活方式,但在 Lifetime Scope 的上下文之外请求实例”
1) 如何在具有 UnitOfWork & Repository 模式的 Windows 窗体中使用 SimpleInjector?
2) 我是否正确地实现了这些模式?
【问题讨论】:
标签: c# entity-framework inversion-of-control repository-pattern simple-injector