18

我想检查一个实体是否有 3 个属性。CreatedDate、ModifiedDate 和 ModifiedBy。

现在我只是在我的对象上下文的 SaveChanges() 方法中对我知道的那些进行硬编码。

例如:

bool newEntity = (entry.State == EntityState.Added);

if (type == typeof(Foo))
{
  var r = entry.Entity as Foo;
  if (r != null)
  {
    if (newEntity) 
      r.CreatedDate = DateTime.Now;
    r.ModifiedDate = DateTime.Now;
    r.ModifiedBy = HttpContext.Current.User.Identity.Name;
  }
}

我知道可以使用类似于以下的代码检查对象是否具有某种方法:

public static bool HasMethod(this object objectToCheck, string methodName)
{
    var type = objectToCheck.GetType();
    return type.GetMethod(methodName) != null;
} 

但是,如果不直接投射实体,我将如何获得这些属性呢?

我该怎么做:

if (HasMethod(entry.Entity))
      entry.Entity.ModifiedDate = DateTime.Now;

我正在使用 ASP.Net MVC 4。

4

3 回答 3

28

您可以使用以下方法。如果存在,它将设置该属性。GetType在每次调用时使用可能会导致一些开销,它需要优化。

private bool TrySetProperty(object obj, string property, object value) {
  var prop = obj.GetType().GetProperty(property, BindingFlags.Public | BindingFlags.Instance);
  if(prop != null && prop.CanWrite) {
    prop.SetValue(obj, value, null);
    return true;
  }
  return false;
}

用法

TrySetProperty(entry.Entity, "ModifiedDate", DateTime.Now);
于 2013-06-06T13:01:06.530 回答
9

您可以使用反射(已经有几个人提到过),也可以创建一个界面。如果你使用自动生成的实体,它们是用partial关键字定义的,所以你可以在同一个项目中创建另一个类文件,并给它相同的命名空间和类定义,你可以让它实现你的接口。然后在上面发布的代码中,检查对象是否实现了您的接口,如果是,则转换为它,然后设置值。

接口的优点是您不使用反射(这可能是一项昂贵的操作),而且,您创建的任何未来实体都将通过实现您的接口自动工作。

如果您的实体属性与您的接口不完全匹配,您可以显式实现该接口,这将处理任何命名异常。

示例:假设您已经定义了一个接口,IContainAuditProperties并且您的所有适用实体都实现了该接口,您可以在循环所有新/更改实体的块内执行以下操作:

var entity = entry.Entity as IContainAuditProperties;
if(entity != null)
{
  entity.CreatedDate = DateTime.Now;
  entity.ModifiedDate = DateTime.Now;
  //etc.
}
于 2013-06-06T13:04:35.930 回答
1

您可以使用以下代码检查对象是否具有具有特定名称的公共可设置属性:

public static bool HasWritableProperty(this object objectToCheck, string propertyName)
{
  var type = objectToCheck.GetType();
  //get a property info for the property, but only if it is a public instance property
  var pi = type.GetProperty(propertyName, BindingFlags.Instance | BindingFlags.Public)
  //return false if no propery is found
  if (pi==null) return false;
  //get the set method for the property
  var setter = pi.GetSetMethod();
  //if it's null the property is not writable
  return (setter != null);
}

然而,这不是一个非常有效的代码,因为它使用了大量的反射。

如果你走这条路,我会记住这段代码的结果,所以每次检查最多运行一次。

我在我的项目中实际所做的(我首先使用 EF 代码)是拥有一个BaseEntity具有常见属性的类,并让具体实体从BaseEntity该类继承。然后我有一个FillEntityMetadata方法,或多或少是这样的:

protected void FillEntityMetadata(BaseEntity entity, bool isUpdate = false)
{
  // Set audit data.
  if (!isUpdate)
  {
    entity.CreatedBy = CurrentUser.ID;
    entity.CreatedOn = DateTime.Now;
    entity.IsActive = true;
  }
  entity.LastModifiedBy = CurrentUser.ID;
  entity.LastModifiedOn = DateTime.Now;
}

请注意,要使其正常工作,您还可以使用接口 ,IBaseEntity它会以相同的方式运行。

于 2013-06-06T13:08:26.007 回答