Entity Framework - 基于外键关联的单向一对一关系

时间:2024-10-17 11:04:02

代码的世界,原以为世界关系很简单,确道是关系无处不在.NET世界里ORM框架中EntityFramework作为其中翘楚,大大解放了搬砖工作的重复工作,着实提高了不少生产力,而也碰到过不少问题!比如关系的映射!

一对一关系的映射:

用户账户密码信息表:包含用户名 密码 邮箱等账户登录时的信息

public class SystemAccount
{
public SystemAccount()
{
Id = DateUtils.GeneratedNewGuid();
}
public Guid Id { get; set; }
public string Salt { get; set; }
public string UserName { get; set; }
public string Password { get; set; }
public string Tele { get; set; }
public Guid UserId { get; set; }
public virtual User User { get; set; }
public virtual User CreateUser(string nick)
{
var user = new User();
user.Nick = nick;
return user; }
}

  用户信息表:包含用户真实姓名,经纬度,昵称,性别等会员信息

public class User
{
public User()
{
Id = DateUtils.GeneratedNewGuid();
}
public Guid Id { get; set; }
public string Nick { get; set; }
public int Gender { get; set; }
public Guid AccountId { get; set; }
public virtual SystemAccount Account { get; set; }
}

  然后fluent api关系配置如下

 public class SystemAccountMapping : FarmerEntityTypeConfiguration<SystemAccount>
{
public SystemAccountMapping()
{
this.HasKey(r => r.Id);
this.Property(r => r.Id).HasDatabaseGeneratedOption(DatabaseGeneratedOption.None);
this.HasRequired(r => r.User).WithMany().HasForeignKey(fk => fk.UserId);
}
}

  

 public class UserMapping : FarmerEntityTypeConfiguration<User>
{
public UserMapping()
{ this.HasKey(r => r.Id);
this.Property(r => r.Id).HasDatabaseGeneratedOption(DatabaseGeneratedOption.None);
this.HasRequired(r => r.Account).WithMany().HasForeignKey(fk => fk.AccountId);
}
}

  然后加上数据库初始化种子数据

internal sealed class CreateIfNoExistDatabase : DbMigrationsConfiguration<FarmerObjectContext>
{
public CreateIfNoExistDatabase()
{
AutomaticMigrationsEnabled = true;
AutomaticMigrationDataLossAllowed = true;
} protected override void Seed(FarmerObjectContext context)
{
SystemAccount account = new SystemAccount();
account.UserName = "shatan776";
account.Salt = "oxml";
account.Password = StringUtils.GenerateSaltedHash("123456", StringUtils.CreateSalt(AppConstants.SaltSize));
account.User = account.CreateUser("老子就是张三");
context.Set<SystemAccount>().Add(account);
context.SaveChanges();
}
}

  

满脸欢喜的debug,来上一口16年的雪碧,静静的等着,结果任务栏宇宙第一ide VisualStudio加蓝,

将 FOREIGN KEY 约束 'FK_dbo.Farmer_User_dbo.Farmer_SystemAccount_AccountId' 引入表 'Farmer_User' 可能会导致循环或多重级联路径。请指定 ON DELETE    NO ACTION 或 ON UPDATE NO ACTION,或修改其他 FOREIGN KEY 约束。
   无法创建约束。请参阅前面的错误消息。

多重循环引用,纳尼,

难道是SystemAccount中加载了User的导航,User中也有,重复了?于是乎,注释掉UserMapping中的导航配置

//this.HasRequired(r => r.Account).WithMany().HasForeignKey(fk => fk.AccountId);

  继续debug错误依据,于是乎又把导航Id注释掉

//public Guid AccountId { get; set; }

哒哒哒达,终于生成了数据库,种子数据也有插入

Entity Framework -  基于外键关联的单向一对一关系

Entity Framework -  基于外键关联的单向一对一关系

一对一的关系就这样生成了,执子之手,与子偕老,张三看上了李四。。。

琢磨着把以前的账户系统移过来,以前的主键是int类型的数据库自增的,修改相应类型后,报错了,找不到account_target,也对,当插入到数据库中时account主键Id是没有生成的,所以user是插入不进去的,笨办法,先插入account,然后再插入user,暂时解决了问题!

看了一下生成的数据库外键图

Entity Framework -  基于外键关联的单向一对一关系

看数据库生成的关系,0=》1。。。可以送SystemAccount实体类中生成User,而User中生成SystemAccount就会报错,用户信息必须有会员账户记录,而会员账户记录可以没有用户信息

更多资料参考

Entity Framework - 理清关系 - 基于外键关联的单向一对一关系