9

我有一个从 POST 请求中获得的模型。由于我的视图定义了它的 POCO 类型,所以从提交的数据创建的对象也是一个 POCO。作为 POCO,它没有覆盖各种虚拟属性。因此,这些虚拟属性返回 null。反过来,这意味着我必须根据外键进行单独的查询以浏览其属性(如果我想做比保存它更复杂的事情)。

鉴于我的模型的 POCO,我能否获得具有所有覆盖功能的代理?

(我曾假设这就是db.Entry().Entity目的,但它仍然返回 POCO 对象,而不是代理。我在断点暂停期间通过鼠标悬停检查对象的运行时类型。)

4

3 回答 3

7

沿着这段代码行的东西会做你需要的。我使用自动映射器将值从传入的实体复制到代理版本。

代码检查传入的实体是否是代理并相应地处理它。

public class Repository<T> where T : class
{
    private readonly Context context;
    private bool mapCreated = false;
    public Repository(Context context)
    {
        this.context = context;
    }

    protected virtual T InsertOrUpdate(T e, int id)
    {
        T instance = context.Set<T>().Create();
        if (e.GetType().Equals(instance.GetType()))
            instance = e;
        else
        {
            if (!mapCreated)
            {
                Mapper.CreateMap(e.GetType(), instance.GetType());
                mapCreated = true;
            }
            instance = Mapper.Map(e, instance);
        }

        if (id == default(int))
            context.Set<T>().Add(instance);
        else
            context.Entry<T>(instance).State = EntityState.Modified;

        return instance;
    }
}

@Colin在不需要自动映射器的评论中描述的更新版本

public class Repository<T> where T : class
{
    private readonly Context context;
    public Repository(Context context)
    {
        this.context = context;
    }

    protected virtual T InsertOrUpdate(T e, int id)
    {
        T instance = context.Set<T>().Create();
        if (e.GetType().Equals(instance.GetType()))
        {
            instance = e;
        }
        else
        {
            DbEntityEntry<T> entry = context.Entry(instance);
            entry.CurrentValues.SetValues(e);
        }

        context.Entry<T>(instance).State =
            id == default(int)
                ? EntityState.Added
                : EntityState.Modified;

        return instance;
    }
}
于 2013-06-18T12:38:32.857 回答
1

db.Entry().Entity 将始终返回一个 POCO,并且不会返回处理虚拟导航属性实现的代理对象:

var o = db.Entry(myPoco).Entity;   // always returns a POCO

Find()在调用或Where()针对数据库上下文时,您通常会得到一个代理对象而不是 POCO 。但是,在对象首次添加到数据库的上下文中,这些方法将(出乎意料?)返回 POCO 而不是代理对象。您实际上必须离开上下文并打开一个新上下文才能获取代理:

        // create a new POCO object, and connect to it to another object already in the DB
        MyPoco myPoco = new MyPoco();
        myPoco.MyOtherPocoId = myPoco2.MyOtherPocoId;   // make reference to existing object

        using (var db = new MyContext())
        {
            // Add myPoco to database.
            db.MyPocos.Add(myPoco);
            db.SaveChanges();

            // One would think you get a proxy object here, but you don't: just a POCO
            var test10 = db.MyPocos.Find(myPoco.Id);                        // returns a MyPoco                        
            var test11 = db.MyPocos.Where(x => x.Id == myPoco.Id).First();  // returns a MyPoco
            var test12 = db.Entry(myPoco).Entity;                           // returns a MyPoco

            // ...so, you can't access the referenced properties through virtual navigation properties:
            MyOtherPoco otherPoco1 = myPoco.Poco2;  // returns NULL
        }

        // leave the context and build a new one

        using (var db = new MyContext())
        {
            // Now, the same Find() and Where() methods return a proxy object
            var test20 = db.MyPocos.Find(myPoco.Id);    // returns a proxy object
            var test21 = db.MyPocos.Where(x => x.Id == myPoco.Id).First();  // returns a proxy object

            // ...which means the virtual properties can be accessed as expected:
            MyOtherPoco otherPoco = myPoco.Poco2;   // works as expected

            // Note that db.Entry().Entity still returns a POCO:
            var test22 = db.Entry(myPoco).Entity;   // returns a MyPoco
        }

可能有一些魔法咒语可以让添加对象的上下文给你一个代理对象,但我还没有遇到过。

于 2013-06-18T13:09:56.310 回答
0

如果您想通过 MVC 控制器执行此操作,您可以使用类似这样的操作:

    [HttpPost]
    public ActionResult Update(int? id, FormCollection form)
    {
        // assumes repository will handle
        // retrieving the entity and
        // including and navigational properties
        var entity = repository.Get(id);
        if (entity == null)
        {
            throw new InvalidOperationException(string.Format("Not found: {0}", id));
        }
        if (TryUpdateModel(entity))
        {
            try
            {
                //
                // do other stuff, additional validation, etc
                repository.Update(entity);
            }
            catch (Exception ex)
            {
                //
                // exception cleansing/handling
                // additional model errors
                return View(entity);
            }
            return View("Success", entity);
        }            

        return View(entity);
    }
于 2013-06-18T13:09:18.160 回答