gpt4 book ai didi

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

转载 作者:太空狗 更新时间:2023-10-29 17:59:30 25 4
gpt4 key购买 nike

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

在我的部分类中,我添加了数据库中不存在的各种只读属性。创建对象时,在我们到达 Controller 的 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; variety,在创建对象时访问。这种行为也会发生在整数属性上

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 实体并通过存储过程/函数导入访问数据库。

我想到的一个想法是,正如 here 所讨论的那样,这些属性作为更改跟踪系统的一部分被访问。似乎正在发生的事情是正在拍摄新创建的项目的快照。我尝试关闭代理创建

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 属性更改为可为 null 的 bool 属性:

public bool? getBoolProperty {
get {
return true;
}
}

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

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

最佳答案

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

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

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

关于c# - ASP.NET MVC 在创建对象时读取属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15546472/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com