ASP.NET MVC在创建对象时读取属性

Mar*_*son 9 c# asp.net-mvc entity-framework

我注意到以下行为,我想知道是否有人可以解释它为什么会发生以及如何防止它.

在我的部分类中,我添加了数据库中不存在的各种只读属性.创建对象时,在访问控制器的Create方法之前访问这些属性.但是,get; set;不会访问使用just声明的属性.

使用非常简单的示例详细说明,如果我有以下属性:

public bool getBoolProperty {
    get {
        return true;
    }
}
public bool getSetBoolProperty {
    get; set; 
}
Run Code Online (Sandbox Code Playgroud)

如果我然后在属性上放置断点,则在创建对象时,将针对第一个属性而不是第二个属性触发断点.但是,如果我有一个方法:

public bool getBoolProperty() {
    return true;
}
Run Code Online (Sandbox Code Playgroud)

那么这是不被访问的.

我尝试了各种各样的变化和注释

// empty set
public bool getBoolProperty {
    get {
        return true;
    }
    set {}
}

// not mapped attribute
[NotMapped]
public bool getBoolProperty {
    get {
        return true;
    }
}

// getting private variable
private bool _boolProperty = true;
public bool getPrivateBoolProperty {
    get {
        return _boolProperty;
    }
}
Run Code Online (Sandbox Code Playgroud)

我已经尝试声明属性virtual,但是get; set;在创建对象时会访问除变体之外的所有变体.整数属性也会发生此行为

public virtual int getIntProperty {
    get {
        return 1;
    }
}
Run Code Online (Sandbox Code Playgroud)

和日期/时间属性,

public virtual DateTime getDateProperty {
    get {
        return DateTime.Now;
    }
}
Run Code Online (Sandbox Code Playgroud)

但不是字符串属性

public string getStringProperty {
    get {
        return "Hello String";
    }
}
Run Code Online (Sandbox Code Playgroud)

或其他实体属性

public Item getItem {
    get {
        return new Item();
    }
}
Run Code Online (Sandbox Code Playgroud)

我遇到的问题是这些属性中的一些可能涉及一些可能需要数据库访问的逻辑,例如

public bool HasChildren {
    get {
        return this.Children !== null && this.Children.Count > 0;
    }
}
Run Code Online (Sandbox Code Playgroud)

在创建时,实体不会拥有.

很显然,我可以做一切的方法解决这个问题,但我很想知道为什么ASP.NET MVC上创建对象(某种可能的内部验证的)访问这些属性,我很想知道,如果有任何防止这种情况的方法,可能通过注释我没有遇到过.

我的项目是数据库优先,C#,EF 4.1,MVC

编辑

我还应该指出我正在使用POCO实体并通过存储过程/函数导入访问数据库.

我想到的一点是,这些属性正在作为此处讨论的变更跟踪系统的一部分进行访问.似乎可能发生的是正在拍摄新创建的项目的快照.我试过关闭代理创建

dbContext.ContextOptions.ProxyCreationEnabled = false;
Run Code Online (Sandbox Code Playgroud)

但是在创建时仍然可以访问这些属性.

编辑20130322

按照@ ladislavmrnka关于探索堆栈跟踪的建议,我得到了这个:

at System.ComponentModel.ReflectPropertyDescriptor.GetValue(Object component)
at System.Web.Mvc.AssociatedMetadataProvider.<>c__DisplayClassb.<GetPropertyValueAccessor>b__a()
at System.Web.Mvc.ModelMetadata.get_Model()
at System.Web.Mvc.DataAnnotationsModelValidator.<Validate>d__1.MoveNext()
at System.Web.Mvc.ModelValidator.CompositeModelValidator.<Validate>d__5.MoveNext()
at System.Web.Mvc.DefaultModelBinder.OnModelUpdated(ControllerContext controllerContext, ModelBindingContext bindingContext)
at System.Web.Mvc.DefaultModelBinder.BindComplexElementalModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Object model)
at System.Web.Mvc.DefaultModelBinder.BindComplexModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
at System.Web.Mvc.DefaultModelBinder.BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
at System.Web.Mvc.ControllerActionInvoker.GetParameterValue(ControllerContext controllerContext, ParameterDescriptor parameterDescriptor)
at System.Web.Mvc.ControllerActionInvoker.GetParameterValues(ControllerContext controllerContext, ActionDescriptor actionDescriptor)
at System.Web.Mvc.ControllerActionInvoker.InvokeAction(ControllerContext controllerContext, String actionName)
at System.Web.Mvc.Controller.ExecuteCore()
at System.Web.Mvc.ControllerBase.Execute(RequestContext requestContext)
at System.Web.Mvc.ControllerBase.System.Web.Mvc.IController.Execute(RequestContext requestContext)
at System.Web.Mvc.MvcHandler.<>c__DisplayClass6.<>c__DisplayClassb.<BeginProcessRequest>b__5()
at System.Web.Mvc.Async.AsyncResultWrapper.<>c__DisplayClass1.<MakeVoidDelegate>b__0()
at System.Web.Mvc.Async.AsyncResultWrapper.<>c__DisplayClass8`1.<BeginSynchronous>b__7(IAsyncResult _)
at System.Web.Mvc.Async.AsyncResultWrapper.WrappedAsyncResult`1.End()
at System.Web.Mvc.MvcHandler.<>c__DisplayClasse.<EndProcessRequest>b__d()
at System.Web.Mvc.SecurityUtil.<GetCallInAppTrustThunk>b__0(Action f)
at System.Web.Mvc.SecurityUtil.ProcessInApplicationTrust(Action action)
at System.Web.Mvc.MvcHandler.EndProcessRequest(IAsyncResult asyncResult)
at System.Web.Mvc.MvcHandler.System.Web.IHttpAsyncHandler.EndProcessRequest(IAsyncResult result)
at System.Web.HttpApplication.CallHandlerExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute()
at System.Web.HttpApplication.ExecuteStep(IExecutionStep step, Boolean& completedSynchronously)
Run Code Online (Sandbox Code Playgroud)

在其中,您将注意到对一两个验证器的调用.作为测试,我将bool属性更改为可以为空的bool属性:

public bool? getBoolProperty {
    get {
        return true;
    }
}
Run Code Online (Sandbox Code Playgroud)

这次创建对象时未访问该属性.这是我想要的行为,但是,我不想将所有自定义属性都更改为可以为空,所以现在我的问题变成......

有没有办法告诉框架不要验证属性? 也许是通过属性. 这个答案几乎回答了这个问题,但由于我首先使用数据库,所以我似乎没有ValidateOnSaveEnabled关闭的属性.也许我需要重新审视我的模型.

Mar*_*son 1

与 EF 的大多数问题一样,答案在于视图模型。

由于在创建模型并将其保存到数据库之前,正在访问的大多数属性都不相关,因此我专门为“创建”视图创建了一个视图模型,该视图仅包含创建模型所需的绝对最小数量的属性。我修改了控制器的 Create 方法以接受我的新视图模型,瞧,在创建这个新模型时,无法访问主模型上的任何不相关属性。

因此,我最初问题的答案似乎是,实体框架在创建对象时对所有不可为空的标量属性执行验证,避免这种情况的一种方法如上所述。