- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
经过大量阅读和尝试后 Entity Framework
最新的稳定版本 (6.1.1)。
我正在阅读很多关于是否使用 EF6
的存储库的矛盾。或 EF
一般来说,因为它是 DbContext
已经提供了一个存储库和 DbSet
UoW
, 盒子外面。
让我首先解释我的解决方案在项目方面包含的内容,然后我将回到矛盾。
它有一个类库项目,还有一个 asp.net-mvc
项目。类 lib 项目是数据访问,其中 Migrations
为 Code First
启用.
在我的类 lib 项目中,我有一个通用存储库:
public interface IRepository<TEntity> where TEntity : class
{
IEnumerable<TEntity> Get();
TEntity GetByID(object id);
void Insert(TEntity entity);
void Delete(object id);
void Update(TEntity entityToUpdate);
}
public class Repository<TEntity> where TEntity : class
{
internal ApplicationDbContext context;
internal DbSet<TEntity> dbSet;
public Repository(ApplicationDbContext context)
{
this.context = context;
this.dbSet = context.Set<TEntity>();
}
public virtual IEnumerable<TEntity> Get()
{
IQueryable<TEntity> query = dbSet;
return query.ToList();
}
public virtual TEntity GetByID(object id)
{
return dbSet.Find(id);
}
public virtual void Insert(TEntity entity)
{
dbSet.Add(entity);
}
public virtual void Delete(object id)
{
TEntity entityToDelete = dbSet.Find(id);
Delete(entityToDelete);
}
public virtual void Update(TEntity entityToUpdate)
{
dbSet.Attach(entityToUpdate);
context.Entry(entityToUpdate).State = EntityState.Modified;
}
}
public DbSet<User> User{ get; set; }
public DbSet<Order> Orders { get; set; }
public DbSet<UserOrder> UserOrders { get; set; }
public DbSet<Shipment> Shipments { get; set; }
EF6
您不再传递存储库,而是传递
DbContext
反而。所以对于
DI
我在
asp-net-mvc
中设置了以下内容项目使用
Ninject
:
private static void RegisterServices(IKernel kernel)
{
kernel.Bind<ApplicationDbContext>().ToSelf().InRequestScope();
}
ApplicationDbContext
在适用的情况下,通过构造函数注入(inject)到上层类。
EF
已经提供了开箱即用的,我们怎么做
Separation of Concern
(在标题中缩写为 SoC)?
asp.net-mvc
项目,如果我不添加存储库。
最佳答案
一些解释有望消除您的困惑。存储库模式的存在是为了抽象出数据库连接和查询逻辑。 ORM(对象关系映射器,如 EF)已经以一种或另一种形式存在了很长时间,以至于许多人已经忘记或从未享受过处理散布着 SQL 查询和语句的意大利面条式代码的巨大乐趣和乐趣。时间是,如果你想查询一个数据库,你实际上要负责一些疯狂的事情,比如启动一个连接和实际从以太构建 SQL 语句。存储库模式的重点是给你一个地方来放置所有这些肮脏的东西,远离你美丽的原始应用程序代码。
快进到 2014 年, Entity Framework 和其他 ORM 是您的存储库。所有的 SQL 逻辑都被整齐地打包,远离你的窥探,取而代之的是你有一个很好的编程 API 可以在你的代码中使用。一方面,这已经足够抽象了。它唯一没有涵盖的是对 ORM 本身的依赖。如果您后来决定要为 NHibernate 甚至 Web API 之类的东西切换出 Entity Framework ,则必须对应用程序进行手术才能这样做。因此,添加另一层抽象仍然是一个好主意,但不是存储库,或者至少可以说是典型的存储库。
您拥有的存储库是一个典型的存储库。它只是为 Entity Framework API 方法创建代理。您拨打 repo.Add
并且存储库调用 context.Add
.坦率地说,这是荒谬的,这就是为什么包括我自己在内的许多人说不要将存储库与 Entity Framework 一起使用。
那你该怎么办?创建服务,或者最好将其称为“类服务类”。当开始讨论与 .NET 相关的服务时,突然间你在谈论与我们在这里讨论的内容完全无关的各种事情。类服务类类似于服务,因为它具有返回特定数据集或对某些数据集执行非常特定功能的端点。例如,对于典型的存储库,您会发现自己在做以下事情:
articleRepo.Get().Where(m => m.Status == PublishStatus.Published && m.PublishDate <= DateTime.Now).OrderByDescending(o => o.PublishDate)
service.GetPublishedArticles();
public interface IService
{
IEnumerable<Article> GetPublishedArticles();
...
}
public class EntityFrameworkService<TContext> : IService
where TContext : DbContext
{
protected readonly TContext context;
public EntityFrameworkService(TContext context)
{
this.context = context;
}
public IEnumerable<Article> GetPublishedArticles()
{
...
}
}
DbSet
直接,即
context.Articles
,但这意味着了解
DbSet
上下文中的名称。最好用
context.Set<TEntity>()
,以获得更大的灵活性。不过,在我跳太多火车之前,我想指出为什么我将其命名为
EntityFrameworkService
.在您的代码中,您只会引用您的
IService
界面。然后,通过您的依赖注入(inject)容器,您可以替换
EntityFrameworkService<YourContext>
为了那个原因。这开启了创建其他服务提供者的能力,比如也许
WebApiService
, 等等。
DbSet
每次通过
var dbSet = context.Set<YourEntity>();
实例.那看起来有点像:
protected virtual IQueryable<TEntity> GetQueryable<TEntity>(
Expression<Func<TEntity, bool>> filter = null,
Func<IQueryable<TEntity>, IOrderedQueryable<TEntity>> orderBy = null,
string includeProperties = null,
int? skip = null,
int? take = null)
where TEntity : class
{
includeProperties = includeProperties ?? string.Empty;
IQueryable<TEntity> query = context.Set<TEntity>();
if (filter != null)
{
query = query.Where(filter);
}
foreach (var includeProperty in includeProperties.Split
(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries))
{
query = query.Include(includeProperty);
}
if (orderBy != null)
{
query = orderBy(query);
}
if (skip.HasValue)
{
query = query.Skip(skip.Value);
}
if (take.HasValue)
{
query = query.Take(take.Value);
}
return query;
}
public IEnumerable<Article> GetPublishedArticles()
{
return GetQueryable<Article>(
m => m.Status == PublishStatus.Published && m.PublishDate <= DateTime.Now,
m => m.OrderByDescending(o => o.PublishDate)
).ToList();
}
public interface IPublishable
{
PublishStatus Status { get; set; }
DateTime PublishDate { get; set; }
}
public IEnumerable<TEntity> GetPublished<TEntity>()
where TEntity : IPublishable
{
return GetQueryable<TEntity>(
m => m.Status == PublishStatus.Published && m.PublishDate <= DateTime.Now,
m => m.OrderByDescending(o => o.PublishDate)
).ToList();
}
service.GetPublished<Article>();
关于c# - EF6 Code First 具有通用存储库和依赖注入(inject)和 SoC,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27527757/
EF POCO 和 EF Code First 有什么区别? 如果我一开始只使用 POCO,我可以将 EF 与它们一起使用吗? 最佳答案 如果您首先使用 EF 代码,您将拥有 POCO 对象,并且数据
EF POCO 和 EF Code First 有什么区别? 如果我一开始只使用 POCO,我可以将 EF 与它们一起使用吗? 最佳答案 如果您首先使用 EF 代码,您将拥有 POCO 对象,并且数据
我有一个基于 .NET 4.8 和 EF 6.4.4 的项目。我们正在逐步迁移到 .Net Core,但在此过程中我可以创建一个 .NET Core 数据上下文类 EF Core 并将两者指向相同的实
我有以下 Entity Framework 5 代码第一类 public class Airplane { public int Id { get; set; } public int
我正在尝试使用 Entity Framework Core 对现有数据库进行逆向工程.我试着按照指示from Microsoft但我遇到了错误: Unable to find provider ass
当数据库不是由 EF 代码首先创建时,有没有办法检查 DbContext 是否与数据库匹配? 我正在寻找与 Database.CompatibleWithModel 类似的功能但没有元数据。 最佳答案
目前,我正在重构我的上下文方法的测试,以不再需要真正的数据库。我使用 Ef Core。 所以我通读了 Microsoft 文档如何测试上下文方法。我首先找到了 EF6 测试的文档,然后阅读了 EfCo
我正在使用 EF 6 Database.SqlQuery 语句来执行存储过程,并实现错误和事务处理,打开和关闭事务(处理多个记录,如果一个记录有错误,仅回滚此,并提交其他任何内容无错误完成)。 Lis
我首先使用 EF 数据库,因为我喜欢在 SQL Management Studio 中设计我的数据库,坦率地说,让 Visual Studio 直接从数据库创建所有实体非常容易,而无需执行任何代码。
我的项目中有几个迁移文件,由于我对上次迁移进行了手动修改,所以我不想使用“程序包管理器控制台”重新生成它。我只需要添加 1 列。所以在之前的迁移中手动添加了这个(我可以这样做,因为还没有人升级过)。
原文:https://bit.ly/2umidlb 作者:jon p smith 翻译:王亮 声明:我翻译技术文章不是逐句翻译的,而是根据我自己的理解来表述的。其中可能会去除一些本人实在不知道如何组
我们想开始一个新项目,我们决定从一开始就在一些表中使用 Columnstore 索引和聚簇索引,我们如何使用 Code First EF Core 3.1 做到这一点? 最佳答案 您应该将主键更改为非
我在 Entity Framework 6.0 上。这是一个开发问题,而不是生产问题。 我想我有一个相互矛盾的策略。 目前,我设置了 DropCreateDatabaseIfModelChanges
我在 VS 2012 RTM 上,使用 EF 5。我在做代码优先,但由于我只是在开发中,所以试图忽略代码迁移。为了避免它们,我有这一套 Database.SetInitializer(new Drop
我有复杂的审计字段类型 我的复杂类型: [ComplexType] public class AuditData { [Column("CreatorUserId")] public
我已经阅读了很多关于如何在关系中指定外键名称的帖子,没有遇到任何问题。不过,我想知道的是,有没有办法更改主键和关系的命名方式? 例如,您有一个以 UserId 作为主键的 User 表。 EF 代码第
我刚刚安装了新的Entity Framework 4.1 NuGet包,因此根据NuGet指令和this article of Scott Hanselman替换了EFCodeFirst包。 现在,想
我的应用程序基于 .NET 4.0 和 EF 4。我现在正在考虑升级到最新版本。 是否有任何可能对我的申请产生不利影响的重大变化或行为差异? 升级路径有多简单?升级到 EF 5 是否需要任何代码更改或
假设您必须使用 EF Code First 和 POCO 类开发支持多种语言的网站,您将如何对 POCO 类进行建模以支持这种情况? 通过多语言支持,我的意思不仅是拥有例如在一些资源文件中为您的 UI
我首先使用 EF 4.1 代码。鉴于以下类片段: public class Doctor { public virtual ICollection Hospitals { get; set;
我是一名优秀的程序员,十分优秀!