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

本文关键字:读取 属性 创建 对象 NET MVC ASP | 更新日期: 2023-09-27 18:05:19

我注意到以下行为,我想知道是否有人可以解释为什么会发生这种情况以及如何预防。

在我的部分类中,我添加了数据库中不存在的各种只读属性。当对象被创建时,这些属性会在我们到达控制器的Create方法之前被访问。但是,仅使用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;
    }
}

我已经尝试声明属性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属性。也许我需要重新审视我的模型。

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

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

因为大多数被访问的属性在模型被创建并持久化到数据库之前是不相关的,所以我为Create视图创建了一个视图模型,它只包含创建模型所需的绝对最小数量的属性。我修改了控制器的Create方法来接受我的新视图模型,瞧,在创建这个新模型时,没有访问主模型上任何不相关的属性。

所以我原来的问题的答案似乎是,实体框架执行验证所有非空标量属性时创建的对象和避免这种情况的一种方法是如上所述。