c# - 关于避免在此抽象上重复代码的任何提示?

标签 c# entity-framework abstract-class data-access-layer

所以我在 SQL Compact Edition 之上使用 Entity Framework 构建了一个应用程序。我不喜欢使用 Entites 作为我的业务对象的想法,所以我一直在构建一个层(我称之为 ObjectModel 层,可能不是最好的术语),它将采用我的普通对象,使用它们填充和保存实体。反之亦然——获取实体,填充 POCO 以用作业务对象


假设我有一个名为 Customer 的 POCO 对象,

public class Customer:ICustomer
    {


        #region ICustomer Members

        public System.Guid id {get;set;}


        public string Forename {get;set;}


        public string Surname { get; set; }


        #endregion
    }

我想使用我的 ObjectModel 层来填充它。我目前这样做...

OM_Customer<ICustomer,Entity.Customer> customerLayer = new OM_Agent<ICustomer,Entity.Customer>();

ICustomer businessObject = customerLayer.GetById(new Guid("4a75d5a5-6f5a-464f-b4b3-e7807806f1a9"));

OM_Customer 类继承自 ObjectModelBase,如下所示。

public abstract class ObjectModelBase<BllClass, EntityClass>
{
    public DataStoreEntities1 db;

    public EntityClass _dataObject;

    public string setName;


    #region POCO-ORM Mapping Functions
    public EntityClass MapBLLToEntity(BllClass bllObject)
    {
        Mapper.CreateMap<BllClass, EntityClass>();
        return Mapper.Map<BllClass, EntityClass>(bllObject);
    }

    public BllClass MapEntityToBLL(EntityClass entityObject)
    {
        Mapper.CreateMap<EntityClass, BllClass>();
        return Mapper.Map<EntityClass, BllClass>(entityObject);
    }
    #endregion

    public void Save(BllClass toAdd)
    {
        _dataObject = MapBLLToEntity(toAdd);

        using (db = new DataStoreEntities1())
        {
            db.AddObject(setName, _dataObject);
            db.SaveChanges();
        }
    }

    public BllClass GetById(Guid id)
    {
        using (db = new DataStoreEntities1())
        {
            EntityClass result = (EntityClass)((object)(from c in  db.CustomerSet  where c.id == id select c).First());
            return MapEntityToBLL(result);
        }

    }

}

这里的问题在于 GetById 方法。子类可以很好地使用 Save 方法,因为所有 Entity Framework 对象上下文都有 AddObject() 方法。

但是,我希望有一个通用的 getById 方法,因为我所有的实体都有一个名为 id 的属性(在数据库中也是如此,作为主键)。问题是这里的 linq...

ModelClass result = (ModelClass)((object)(from c in  db.CustomerSet  where c.id == id select c).First());

特定于 db.CustomerSet。我希望这个抽象类为我的所有实体提供通用的 CRUD/Paging/'Collection Retrieval',但没有重复的击键,但是,这个简单的 GetById 方法促使我寻求帮助。

当前想法:

  • 让我通过 Id 使用 Entity SQL,这样我就可以更好地控制查询字符串的构建。
  • 废弃 Entity Framework 和 linq,因为完全抽象可能太耗时。

对于 SO 社区将如何解决上述问题的任何建议,我们将不胜感激。

最佳答案

首先,我会首先质疑为什么您认为使用 EF 实体作为业务实体必然是一件坏事?我已经在许多项目上这样做了,但我还没有遇到任何我无法克服的严重问题。

话虽如此,如果您真的觉得有充分的理由将两者分开,这是一个选择。

您可以使用一个委托(delegate)来注入(inject)要查询的特定实体集对象,另一个用于键的选择器——它可以允许派生对象仅指定它们的存储语义。所以你的代码看起来像:

public abstract class ObjectModelBase<BllClass, EntityClass>
{
    // ... same basic code here...

    // supplied by your derivatives...
    protected abstract Func<IQueryable<EntityClass>> GetEntitySet { get; };
    protected abstract Func<Guid,Func<EntityClass,bool>> KeySelector { get; }

    public BllClass GetById(Guid id)
    {
        using (db = new DataStoreEntities1())
        {
            EntityClass result = (EntityClass)((object)
                 GetEntitySet()
                     .Where( KeySelector( id ) )
                     .First();
            return MapEntityToBLL(result);
        }
    }
}

public class OM_Customer : ObjectModelBase<ICustomer,Entity.Customer>
{
    protected abstract Func<IQueryable<Entity.Customer>> GetEntitySet
    { 
        get { return db.CustomerSet; }
    }

    protected abstract Func<Guid,Func<Entity.Customer,bool>> KeySelector
    {
        get { return (g => (e => e.Id == g)); }
    }
}

关于c# - 关于避免在此抽象上重复代码的任何提示?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1089580/

相关文章:

c# - 在 C# 中将 float 转换为十进制。为什么 (decimal)0.1F == 0.1M 不是因为四舍五入而为假?

c# - Nancy:我如何捕获所有请求,而不考虑动词或路径

c# - Entity Framework 绕过 QueryFilter/SoftDelete

c# - 如何使用方法实现基类并强制派生类覆盖它?

c++ - 实现抽象类的类中的多态函数

c# - 使用 Lambda 表达式选择类方法名称

c# - dropdownlistfor 在 'for' 循环内不发布值

c# - ASP.NET MVC 搜索页面 - Linq + EF4 上的整数 StartsWith

.net - Entity Framework 无法先生成数据库 EDMX

java - 我们是否需要在命名约定方面区分接口(interface)类和抽象类?