【问题标题】:How to determine if a PropertyType is foreign key如何确定 PropertyType 是否为外键
【发布时间】:2014-04-04 09:54:46
【问题描述】:

我有以下类“schakeling”,由 EF 生成,代表数据库表“schakeling”。在数据库中,'id' 是主键,'plc_id' 是外键。

public partial class schakeling
{
    public schakeling()
    {
        this.verbruik = new HashSet<verbruik>();
    }

    public int id { get; set; }
    public int plc_id { get; set; }
    public string var_output { get; set; }
    public string omschrijving { get; set; }
    public int vermogen { get; set; }
    public Nullable<bool> status { get; set; }
    public Nullable<byte> hourOn { get; set; }
    public Nullable<byte> minuteOn { get; set; }
    public Nullable<byte> hourOff { get; set; }
    public Nullable<byte> minuteOff { get; set; }

    public virtual plc plc { get; set; }
    public virtual ICollection<verbruik> verbruik { get; set; }
}

我有一个视图类

public class SchakelingsListViewModel
{
    public IEnumerable<schakeling> List { get; set; }
    public PagingInfo PagingInfo { get; set; }//Not relevant for this question
    //And some more properties...also not relevant
}

我有以下视图(为简洁起见,省略了一些 HTML)

@model PortalControl.ViewModels.SchakelingsListViewModel
<h2>Index</h2>
<table>
@foreach (var item in Model.List) {
    @Html.TableRowFor(item)
}
</table>

我有一个通用的 html 辅助方法 TableRowFor,因为我希望能够在使用 EF 生成的其他域实体上使用该方法。该方法生成简单的表数据。

public static MvcHtmlString TableRowFor<T>(this HtmlHelper helper, T obj)
{
    string controller = obj.GetType().BaseType.Name;
    string id = obj.GetType().GetProperty("id").GetValue(obj).ToString();

    StringBuilder sb = new StringBuilder("<tr>");
    sb.Append("<td>");
    sb.Append("<a href='" + controller + "/Edit/" + id + "'><img src='/Images/edit-icon.png' /></a>");
    sb.Append("<a href='" + controller + "/Details/" + id + "'><img src='/Images/details-icon.png' /></a>");
    sb.Append("<a href='" + controller + "/Delete/" + id + "'><img src='/Images/delete-icon.png' /></a>");
    sb.Append("</td>");

    foreach (var property in obj.GetType().GetProperties())
    {
        //If statement below filters out the two virtual properties(plc, verbruik) of the schakeling class(as said, generated with EF), somewhat ugly but it works, better suggestions are welcome..
        if ((!property.PropertyType.Name.ToLower().Contains("icollection")) && (property.PropertyType.CustomAttributes.Count() != 0))
        {
            sb.Append("<td>");
            //if property is foreign key
                //sb.Append("<a href='" + correctControllerNameHere + "/Details/" + property.GetValue(obj) + "'><img src='/Images/details-icon.png' /></a>");
            //else
                //sb.Append(property.GetValue(obj));
            sb.Append("</td>");
        }
    }
    sb.Append("</tr>");
    return new MvcHtmlString(sb.ToString());
}

我的问题是,如果属性是外键,我想创建一个链接。

我在互联网上搜索过,但我不是 PropertyInfo、MetaDataClassType 和其他类似类的专家。像 property.isForeign() 这样的东西会很可爱,但任何有效的东西都会受到赞赏。

【问题讨论】:

    标签: c# asp.net-mvc entity-framework foreign-keys html-helper


    【解决方案1】:

    您可以通过此方法从 Entity Framework 概念模型中获取参考导航属性:

    IEnumerable<string> GetReferenceProperies<T>(DbContext context)
    {
        var oc = ((IObjectContextAdapter)context).ObjectContext;
        var entityType = oc.MetadataWorkspace.GetItems(DataSpace.OSpace)
                           .OfType<EntityType>()
                           .FirstOrDefault (et => et.Name == typeof(T).Name);
        if (entityType != null)
        {
            foreach (NavigationProperty np in entityType.NavigationProperties
                    .Where(p => p.ToEndMember.RelationshipMultiplicity
                                         == RelationshipMultiplicity.One
                             || p.ToEndMember.RelationshipMultiplicity
                                         == RelationshipMultiplicity.ZeroOrOne))
            {
                yield return np.Name;
            }
        }
    }
    

    它获取所有在关联末尾具有0..1 的导航属性,因此不包括集合导航属性。

    现在您可以使用属性名称来获取匹配的PropertyInfos 并获取属性的值。

    【讨论】:

      【解决方案2】:

      我用你的方法搞定了,非常感谢。对于其他人,请记住,此解决方案仅在您使用像我这样的命名约定时才有效(但是,您可以使用 Gert 的方法作为指导,为您自己的解决方案铺平道路):

      • 为外键指定与其引用的数据库表名称完全相同的名称
      • 名称以“_id”结尾

      例子:

      您有一个名为 user 的 db 表,并希望创建一个名为 bill 的新 db 表。您想向使用 FK 的用户提交账单,因此在我的情况下,我将命名 FK user_id

      更新了 TableRowFor 方法:

      public static MvcHtmlString TableRowFor<T>(this HtmlHelper helper, T obj)
      {
          string controller = obj.GetType().BaseType.Name;
          string id = obj.GetType().GetProperty("id").GetValue(obj).ToString();
      
          StringBuilder sb = new StringBuilder("<tr>");
          sb.Append("<td>");
          sb.Append("<a href='" + controller + "/Edit/" + id + "'><img src='/Images/edit-icon.png' /></a>");
          sb.Append("<a href='" + controller + "/Details/" + id + "'><img src='/Images/details-icon.png' /></a>");
          sb.Append("<a href='" + controller + "/Delete/" + id + "'><img src='/Images/delete-icon.png' /></a>");
          sb.Append("</td>");
      
          List<string> referencePropertyList = GetReferenceProperies<T>(new NameOfDB()).ToList();
          foreach (var property in obj.GetType().GetProperties())
          {
              //If statement below filters out the two virtual properties(plc, verbruik) of the schakeling class(as said, generated with EF), somewhat ugly but it works, better suggestions are welcome..
              if ((!property.PropertyType.Name.ToLower().Contains("icollection")) && (property.PropertyType.CustomAttributes.Count() != 0))
              {
                  sb.Append("<td>");
                  //if property is foreign key
                  if (referencePropertyList != null && property.Name.Length >= 3 && referencePropertyList.Contains(property.Name.Substring(0, property.Name.Length - 3)))
                      sb.Append("<a href='" + property.Name.Substring(0, property.Name.Length - 3 ) + "/Details/" + property.GetValue(obj) + "'>" + property.GetValue(obj) + "</a>");
                  else
                      sb.Append(property.GetValue(obj));
                  sb.Append("</td>");
              }
          }
          sb.Append("</tr>");
          return new MvcHtmlString(sb.ToString());
      }
      

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 2021-07-21
        • 2018-06-25
        • 1970-01-01
        • 2011-06-08
        • 1970-01-01
        • 2010-10-16
        • 2021-11-01
        • 1970-01-01
        相关资源
        最近更新 更多