自定义导航属性

时间:2014-12-16 21:41:37

标签: c# entity-framework entity-framework-6

假设我们在数据库中有一个表 Car ,如下所示:

Id | Brand | Model | Color | Description
 1 |  23   |  6    |  005  | Beautiful car

另一个名为元数据的表格,其中包含有关汽车的不同信息。

Id | Type  | Key | Value
 1 | Brand |   6 | Ford
 2 | Brand |  22 | BMW
 3 | Brand |  23 | Audi
 4 | Model |   5 | Focus
 5 | Model |   6 | A4
 6 | Model |   7 | 325
 7 | Color | 005 | Black
 8 | Color | 019 | Blue

如您所见,组合类型&密钥在表中应该是唯一的,可以被认为是外键。

我完全理解规范化数据库的概念。在我们的情况下,对Metadata表进行规范化会在应用程序的其他部分引入复杂性,这是我们能够进入的原因。

Code First类CarMetadata看起来像这样

public class Car
{
    public int Id { get; set; }
    public string BrandId { get; set; }
    public string ModelId { get; set; }
    public string ColorId { get; set; }
}

public class Metadata
{
    public int Id { get; set; }
    public string Type { get; set; }
    public string Key { get; set; }
    public string Value { get; set; }
}

CarMetadata

的配置类
public class CarMap : EntityTypeConfiguration<Car>
{
    this.HasKey(t => t.Id);
    this.Property(t => t.BrandId).IsRequired().HasMaxLength(6);
    this.Property(t => t.ModelId).IsRequired().HasMaxLength(6);
    this.Property(t => t.ColorId).IsRequired().HasMaxLength(6);

    this.ToTable("Car");
    this.Property(t => t.Id).HasColumnName("Id");
    this.Property(t => t.BrandId).HasColumnName("Brand");
    this.Property(t => t.ModelId).HasColumnName("Model");
    this.Property(t => t.ColorId).HasColumnName("Color");
}

public class MetadataMap : EntityTypeConfiguration<Metadata>
{
    this.HasKey(t => t.Id);
    this.Property(t => t.Type).IsRequired().HasMaxLength(6);
    this.Property(t => t.Key).IsRequired().HasMaxLength(6);
    this.Property(t => t.Value).IsRequired().HasMaxLength(6);

    this.ToTable("Metadata");
    this.Property(t => t.Id).HasColumnName("Id");
    this.Property(t => t.Type).HasColumnName("Type");
    this.Property(t => t.Key).HasColumnName("Key");
    this.Property(t => t.Value).HasColumnName("Value");
}

DbContext

public class CarContext : DbContext
{
    public DbSet<Car> Cars { get; set; }
    public DbSet<Metadata> Metadata { get; set; }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Configurations.Add(new CarMap());
        modelBuilder.Configurations.Add(new MetadataMap());
    }
}

检索前10辆车的方法

public List<Car> GetAllCars()
{
    using (CarContext context = new CarContext())
    {
        return context.Cars.ToList();
    }
}

假设我将导航属性添加到Car

public class Car
{
    public int Id { get; set; }

    public string BrandId { get; set; }
    public virtual Metadata Brand { get; set; }

    public string ModelId { get; set; }
    public virtual Metadata Model { get; set; }

    public string ColorId { get; set; }
    public virtual Metadata Color { get; set; }
} 

在调用 GetAllCars()时,确保实现这些属性的最优雅,最有效的方法是什么?

1 个答案:

答案 0 :(得分:0)

在这种情况下,我认为你不能拥有“真正的”导航属性,因为我没有看到你如何在数据库中定义这种关系和/或使用流畅的API在代码中映射它。 (有人可能会在这里纠正我)。

在这种情况下,我会将这些定义为未映射,并在GetAllCars()方法中填充它们。所以,Car类看起来像这样:

public class Car
{
    public int Id { get; set; }

    public string BrandId { get; set; }
    [NotMapped]
    public Metadata Brand { get; set; }

    public string ModelId { get; set; }
    [NotMapped]
    public Metadata Model { get; set; }

    public string ColorId { get; set; }
    [NotMapped]
    public Metadata Color { get; set; }
}

而且,方法是:

public List<Car> GetAllCars()
{
    using (CarContext context = new CarContext())
    {
        var returnList = new List<Car>();

        foreach(var car in context.Cars)
        {
            car.Brand = context.Metadata
                               .Where(x => x.Key == car.BrandId && x.Type == "Brand")
                               .FirstOrDefault();

            car.Model = context.Metadata
                               .Where(x => x.Key == car.ModelId && x.Type == "Model")
                               .FirstOrDefault();

            car.Color = context.Metadata
                               .Where(x => x.Key == car.ColorId && x.Type == "Color")
                               .FirstOrDefault();

            returnList.Add(car);
        }

        return returnList;
    }
}