使用代码映射通过 Session.Save 通过 NHibernate 4 插入 SQL 服务器数据库

Insert into SQL Server database through NHibernate 4 with Session.Save using mapping-by-code

当前遇到 Session.Save() 未插入记录并产生以下异常的问题:

        null id in NHModels.Domain.Activity entry (don't flush the Session after an exception occurs)

at NHibernate.Event.Default.DefaultFlushEntityEventListener.CheckId(Object obj, IEntityPersister persister, Object id, EntityMode entityMode)
at NHibernate.Event.Default.DefaultFlushEntityEventListener.GetValues(Object entity, EntityEntry entry, EntityMode entityMode, Boolean mightBeDirty, ISessionImplementor session)
at NHibernate.Event.Default.DefaultFlushEntityEventListener.OnFlushEntity(FlushEntityEvent event)
at NHibernate.Event.Default.AbstractFlushingEventListener.FlushEntities(FlushEvent event)
at NHibernate.Event.Default.AbstractFlushingEventListener.FlushEverythingToExecutions(FlushEvent event)
at NHibernate.Event.Default.DefaultFlushEventListener.OnFlush(FlushEvent event)
at NHibernate.Impl.SessionImpl.Flush()
at NHibernate.Transaction.AdoTransaction.Commit()
at NHUnitOfWork.Dispose() in NHUnitOfWork.cs:line
at DatabaseActivityOperations.<WriteActivity>d__6.MoveNext() in DatabaseActivityOperations.cs:line 240

我的 class 和映射看起来像这样。

Activity(为简单起见,我删除了几个与此相关的 ILists class)

public class Activity {

    public Activity() {
        Activityschema = new List<ActivitySchema>();
    }

    public virtual int ActivityKey { get; set; }
    public virtual string Activityname { get; set; }
    public virtual string Activitydescription { get; set; }
    public virtual DateTime Averageactivitytime { get; set; }
    public virtual int Averagenumberpeople { get; set; }
    public virtual string Worktype { get; set; }
    public virtual bool? Canautocomplete { get; set; }
    public virtual IList<ActivitySchema> Activityschema { get; set; }
}

Activity地图

public class ActivityMap : ClassMapping<Activity> {

    public ActivityMap() {
        Schema("dbo");
        Lazy(true);
        Id(x => x.ActivityKey, map => { map.Generator(Generators.Identity); });
        Property(x => x.Activityname, map => { map.NotNullable(true); map.Length(50); });
        Property(x => x.Activitydescription, map => { map.NotNullable(true); map.Length(100); });
        Property(x => x.Averageactivitytime, map =>
        {
            map.NotNullable(true);
            map.Type(NHibernateUtil.Time);
        });
        Property(x => x.Averagenumberpeople, map => { map.NotNullable(true); map.Precision(10); });
        Property(x => x.Worktype, map => { map.NotNullable(true); map.Length(50); });
        Property(x => x.Canautocomplete);
        Bag(x => x.Activityschema, colmap =>  { colmap.Key(x => x.Column("ActivityKey")); colmap.Inverse(true); }, map => { map.OneToMany(); }); 
    }
}

最后,这是工作单元 class 我有:

public class NHUnitOfWork : IDisposable
{
    public static string ConnectingString { get; private set; } = @"data source=nh;initial catalog=db;MultipleActiveResultSets=True;";
    protected static Configuration _config;
    protected static NHibernate.ISessionFactory _sessionFactory;
    public NHibernate.ISession Session { get; private set; }
    protected NHibernate.ITransaction Transaction { get; set; }
    private const System.Data.IsolationLevel ISOLATION_LEVEL = System.Data.IsolationLevel.ReadUncommitted;
    private bool RollBack { get; set; } = false;

    public NHUnitOfWork(string databaseConnectionString)
    {
        if (_config == null)
        {
            var cfg = new Configuration();
            cfg.DataBaseIntegration(db =>
            {
                db.Driver<NHibernate.Driver.SqlClientDriver>();
                db.ConnectionString = @"data source=nh;initial catalog=db;MultipleActiveResultSets=True;";
                //db.ConnectionString = databaseConnectionString;
                db.Dialect<MsSql2012Dialect>();
                db.BatchSize = 500;
            })
            .AddAssembly(typeof(Activity).Assembly)
            .SessionFactory()
            .GenerateStatistics();

            var mapper = new ModelMapper();
            mapper.AddMappings(typeof(ActivityMap).Assembly.GetTypes());
            cfg.AddMapping(mapper.CompileMappingForAllExplicitlyAddedEntities());
            _config = cfg;
            _sessionFactory = _config.BuildSessionFactory();
        }
        Session = _sessionFactory.OpenSession();
        Transaction = Session.BeginTransaction(ISOLATION_LEVEL);
        RollBack = false;
    }

    public void Commit()
    {
        Transaction.Commit();
    }

    public void Rollback()
    {
        if (Transaction.IsActive) Transaction.Rollback();
    }

    public void Dispose()
    {
        if (RollBack)
        {
            Transaction.Rollback();
        }
        else
        {
            Transaction.Commit();
        }

        Session.Close();
    }
}

至此,我相信我的配置是正确的。我已成功使用 Session.Query 读取数据并且没有出现问题。当我写这样的东西来添加新记录时,问题就来了:

    var activity = new Activity
    {
        Activityname = "TestActivity",
        Activitydescription = "This is a test",
        Averagenumberpeople = 1,
        Worktype = "Test",
        Canautocomplete = false,
        Averageactivitytime = new DateTime(1, 1, 1, 0, 55, 55)
    };
    using (var uow = new NHUnitOfWork(NHUnitOfWork.ConnectingString))
    {
        uow.Session.Save(activity); // Produces exception here

        //This also produces an exception
        //uow.Session.Save(activity, Generators.Identity);
    }

我认为这与我如何在 ActivityMap 中映射 ID 以及生成器未按预期工作有关。我试图将它更改为其他几种类型并得到相同的异常,或者一个声明它无法转换为 SystemInt32。我也试过将 ID 更改为 long 并指定数据类型,但没有成功。我在这里似乎做错了什么?

在我发布这篇文章几分钟后,我发现问题实际上出在我设置日期时间的方式上。

new DateTime(1, 1, 1, 0, 55, 55)

它不喜欢“1/1/0001”部分,所以这似乎是导致问题的原因,我只关心时间片。将年份更改为 2001 年之类的内容修复了插入内容并且它工作正常,消息描述性不强。