我注意到以下行为,我想知道是否有人可以解释它为什么会发生以及如何防止它。
在我的部分类中,我添加了数据库中不存在的各种只读属性。创建对象时,在我们访问控制器的 Create 方法之前会访问这些属性。但是,使用 just 声明的属性get; set;
不会被访问。
如果我具有以下属性,请使用非常简单的示例进行详细说明:
public bool getBoolProperty {
get {
return true;
}
}
public bool getSetBoolProperty {
get; set;
}
如果我然后在属性上放置断点,则在创建对象时会针对第一个属性而不是第二个属性命中断点。但是,如果我有一个方法:
public bool getBoolProperty() {
return true;
}
那么这不会被访问。
我尝试了各种变体和注释
// 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;
}
}
我尝试过声明 properties virtual
,但除了get; set;
品种之外的所有变体都在创建对象时被访问。整数属性也会出现此行为
public virtual int getIntProperty {
get {
return 1;
}
}
和日期/时间属性,
public virtual DateTime getDateProperty {
get {
return DateTime.Now;
}
}
但不适用于字符串属性
public string getStringProperty {
get {
return "Hello String";
}
}
或其他实体属性
public Item getItem {
get {
return new Item();
}
}
我遇到的问题是,其中一些属性可能涉及一些可能需要数据库访问的逻辑,例如
public bool HasChildren {
get {
return this.Children !== null && this.Children.Count > 0;
}
}
在创建时,实体将没有。
显然,我可以通过将所有内容都设为方法来解决这个问题,但我很想知道为什么 ASP.NET MVC 在创建对象时访问这些属性(可能是某种内部验证),并且我很想知道是否有任何防止这种情况的方法,可能是通过我没有遇到的注释。
我的项目是数据库优先,C#,EF 4.1,MVC
编辑
我还应该指出我正在使用 POCO 实体并通过存储过程/函数导入来访问数据库。
我想到的一个想法是,这些属性正在作为此处讨论的更改跟踪系统的一部分进行访问。似乎正在发生的可能是正在为新创建的项目拍摄快照。我尝试关闭代理创建
dbContext.ContextOptions.ProxyCreationEnabled = false;
但是在创建时仍然可以访问这些属性。
编辑 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)
在其中,您会注意到对一两个验证器的调用。作为测试,我将 bool 属性更改为可为空的 bool 属性:
public bool? getBoolProperty {
get {
return true;
}
}
这次在创建对象时没有访问该属性。这是我想要的行为,但是,我不想将我的所有自定义属性都更改为可以为空,所以我的问题现在变成了......
有没有办法告诉框架不要验证属性? 也许通过一个属性。 这个答案几乎回答了这个问题,但由于我首先使用数据库,我似乎没有ValidateOnSaveEnabled
要关闭的属性。也许我需要重新审视我的模型。