Col*_*lin 31 entity-framework data-annotations ef-code-first
我有2个课程:客户和调查.
每个客户可以进行多次调查 - 但只有一次默认调查.
我已经定义了这样的类:
public class Client
{
    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int ID { get; set; }
    public string ClientName { get; set; }
    public Nullable<int> DefaultSurveyID { get; set; }
    [ForeignKey("DefaultSurveyID")]
    public virtual Survey DefaultSurvey { get; set; }
    public virtual ICollection<Survey> Surveys { get; set; }
}
public class Survey
{
    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int ID { get; set; }
    public string SurveyName { get; set; }
    [Required]
    public int ClientID { get; set; }
    [ForeignKey("ClientID")]
    public virtual Client Client { get; set; }
}
这会像我期望的那样创建Client表:
[dbo].[Clients]
(
[ID] [int] IDENTITY(1,1) NOT NULL,
[ClientName] [nvarchar](max) NULL,
[DefaultSurveyID] [int] NULL
)
但Survey表有一个额外的外键:
[dbo].[Surveys]
(
[ID] [int] IDENTITY(1,1) NOT NULL,
[SurveyName] [nvarchar](max) NULL,
[ClientID] [int] NOT NULL,
[Client_ID] [int] NULL
)
为什么Code First会产生这种关系以及如何告诉它不要?
eca*_*ver 42
问题是当你在两个实体之间有多个关系时,EF Code First无法找出哪些导航属性匹配,除非你告诉它如何,这里是代码:
public class Client
{
    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int ID { get; set; }
    public string ClientName { get; set; }
    /****Change Nullable<int> by int?, looks better****/
    public int? DefaultSurveyID { get; set; }
    /****You need to add this attribute****/
    [InverseProperty("ID")]
    [ForeignKey("DefaultSurveyID")]
    public virtual Survey DefaultSurvey { get; set; }
    public virtual ICollection<Survey> Surveys { get; set; }
}
前一版本,EF是创造额外的关系,因为它不知道该DefaultSurvey财产被引用ID的的Survey类,但你可以让它知道,添加属性InverseProperty,其参数是属性的名称Survey,你需要DefaultSurvey到配合.
Phi*_*hil 11
你可以使用代码优先,但不是我欺骗的代码第一专家:-)
1)我使用SMS在数据库中创建了表和关系(如上所述,没有额外的Client_ID)
2)我使用逆向工程代码优先创建所需的类和映射
3)我删除了数据库并使用context.Database.Create()重新创建它.
原始表定义:
CREATE TABLE [dbo].[Client](
    [Id] [int] IDENTITY(1,1) NOT NULL,
    [Name] [nvarchar](50) NULL,
    [DefaultSurveyId] [int] NULL,
     CONSTRAINT [PK_dbo.Client] PRIMARY KEY NONCLUSTERED 
    (
        [Id] ASC
    )
)
CREATE TABLE [dbo].[Survey](
    [Id] [int] IDENTITY(1,1) NOT NULL,
    [Name] [nvarchar](50) NULL,
    [ClientId] [int] NULL,
     CONSTRAINT [PK_dbo.Survey] PRIMARY KEY NONCLUSTERED 
    (
        [Id] ASC
    )
)
加上外键
ALTER TABLE [dbo].[Survey]  WITH CHECK 
    ADD CONSTRAINT [FK_dbo.Survey_dbo.Client_ClientId] FOREIGN KEY([ClientId])
    REFERENCES [dbo].[Client] ([Id])
ALTER TABLE [dbo].[Client]  WITH CHECK 
    ADD CONSTRAINT [FK_dbo.Client_dbo.Survey_DefaultSurveyId] 
    FOREIGN KEY([DefaultSurveyId]) REFERENCES [dbo].[Survey] ([Id])
逆向工程生成的代码:
public partial class Client
{
    public Client()
    {
        this.Surveys = new List<Survey>();
    }
    public int Id { get; set; }
    public string Name { get; set; }
    public int? DefaultSurveyId { get; set; }
    public virtual Survey DefaultSurvey { get; set; }
    public virtual ICollection<Survey> Surveys { get; set; }
}
public partial class Survey
{
    public Survey()
    {
        this.Clients = new List<Client>();
    }
    public int Id { get; set; }
    public string Name { get; set; }
    public int? ClientId { get; set; }
    public virtual ICollection<Client> Clients { get; set; }
    public virtual Client Client { get; set; }
}
public class ClientMap : EntityTypeConfiguration<Client>
{
    #region Constructors and Destructors
    public ClientMap()
    {
        // Primary Key
        this.HasKey(t => t.Id);
        // Properties
        this.Property(t => t.Name).HasMaxLength(50);
        // Table & Column Mappings
        this.ToTable("Client");
        this.Property(t => t.Id).HasColumnName("Id");
        this.Property(t => t.Name).HasColumnName("Name");
        this.Property(t => t.DefaultSurveyId).HasColumnName("DefaultSurveyId");
        // Relationships
        this.HasOptional(t => t.DefaultSurvey)
            .WithMany(t => t.Clients).HasForeignKey(d => d.DefaultSurveyId);
    }
    #endregion
}
public class SurveyMap : EntityTypeConfiguration<Survey>
{
    #region Constructors and Destructors
    public SurveyMap()
    {
        // Primary Key
        this.HasKey(t => t.Id);
        // Properties
        this.Property(t => t.Name).HasMaxLength(50);
        // Table & Column Mappings
        this.ToTable("Survey");
        this.Property(t => t.Id).HasColumnName("Id");
        this.Property(t => t.Name).HasColumnName("Name");
        this.Property(t => t.ClientId).HasColumnName("ClientId");
        // Relationships
        this.HasOptional(t => t.Client)
            .WithMany(t => t.Surveys).HasForeignKey(d => d.ClientId);
    }
    #endregion
}
| 归档时间: | 
 | 
| 查看次数: | 58108 次 | 
| 最近记录: |