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

  代码的世界,原以为世界关系很简单,确道是关系无处不在.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; }

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

 

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

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

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

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

更多资料参考

 

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

posted @ 2016-08-06 00:17  沙滩裤  阅读(725)  评论(0编辑  收藏  举报