0

私は NHibernate を初めて使用し、Automap 機能を適切に動作させるためにいくつかの問題に直面しています。ここに私が抱えているいくつかの問題があります。

Fluent NHibernate の入門 wiki ( http://wiki.fluentnhibernate.org/Getting_started)では、店舗、製品、および従業員クラスのサンプルと、それらのクラスのマッピングが定義されています。手動マッピングを AutoMapping に置き換え、Fluent NHibernate を使用してスキーマを生成しました。すべてが適切に生成されました。ただし、アプリケーションがサンプル ストア、製品、および従業員オブジェクトを保存しようとすると、「TransientObjectException was Unhandled: オブジェクトは保存されていない一時インスタンスを参照しています - フラッシュする前に一時インスタンスを保存します。タイプ: FluentExample.Entities.Employee、エンティティ: FluentExample.Entities.Employee" .

自動マップは次のようになります。

.Mappings(m=>
  m.AutoMappings.Add(
    AutoMap.AssemblyOf<FluentExample.Entities.Employee>(type => type.Namespace == "FluentExample.Entities")))

オブジェクト作成コード (wiki からそのまま引用) は次のようになります。手動の流暢なマッピングを使用すると、オブジェクトの作成が正常に機能することに注意してください。

// create a couple of Stores each with some Products and Employees
var barginBasin = new Store { Name = "Bargin Basin" };
var superMart = new Store { Name = "SuperMart" };

var potatoes = new Product { Name = "Potatoes", Price = 3.60 };
var fish = new Product { Name = "Fish", Price = 4.49 };
var milk = new Product { Name = "Milk", Price = 0.79 };
var bread = new Product { Name = "Bread", Price = 1.29 };
var cheese = new Product { Name = "Cheese", Price = 2.10 };
var waffles = new Product { Name = "Waffles", Price = 2.41 };

var daisy = new Employee { FirstName = "Daisy", LastName = "Harrison" };
var jack = new Employee { FirstName = "Jack", LastName = "Torrance" };
var sue = new Employee { FirstName = "Sue", LastName = "Walkters" };
var bill = new Employee { FirstName = "Bill", LastName = "Taft" };
var joan = new Employee { FirstName = "Joan", LastName = "Pope" };

// add products to the stores, there's some crossover in the products in each
// store, because the store-product relationship is many-to-many
AddProductsToStore(barginBasin, potatoes, fish, milk, bread, cheese);
AddProductsToStore(superMart, bread, cheese, waffles);

// add employees to the stores, this relationship is a one-to-many, so one
// employee can only work at one store at a time                     
AddEmployeesToStore(barginBasin, daisy, jack, sue);
AddEmployeesToStore(superMart, bill, joan);

// save both stores, this saves everything else via cascading
session.SaveOrUpdate(barginBasin);
session.SaveOrUpdate(superMart);

transaction.Commit();

自分のクラスの 1 つで AutoMap 機能を使用しようとすると、クラスが作成されますが、実際にレコードを挿入しようとすると、何らかの理由でエラーが発生します。主なエラー メッセージには、「AssertionFailure は処理されませんでした: null 値」と表示されます。これは、私のクラス、構成/マッピング、エラー、およびテーブル作成スクリプトのサンプルです。(注: クラスの属性は ASP.NET MVC で使用するためのものであり、NH とは関係ありません。)

namespace Credit.Data.Entities
{
  [Serializable]
  public class EthnicityType
  {
    public EthnicityType()
    {
    }

    [DisplayName("Id")]
    [Required(ErrorMessage = "Id is required.")]
    public virtual Guid Id { get; private set; }

    [DisplayName("Title")]
    [Required(ErrorMessage = "Title is required.")]
    [StringLength(80, ErrorMessage = "Title must be less than 80 characters.")]
    public virtual string Title { get; set; }

    [DisplayName("Description")]
    [StringLength(255, ErrorMessage = "Description must be less than 255 characters.")]
    public virtual string Description { get; set; }

    [DisplayName("Is Active")]
    [Required(ErrorMessage = "Is Active is required.")]
    public virtual bool IsActive { get; set; }
  }
}

Fluent NHibernate の構成は次のとおりです。

private static ISessionFactory CreateSessionFactory()
{
  return Fluently.Configure()
    .Database(MsSqlConfiguration.MsSql2008
      .ConnectionString(c => c.FromConnectionStringWithKey ("CreditConnectionString"))
      .UseReflectionOptimizer()
      .AdoNetBatchSize(25)
      .DefaultSchema("dbo")
      .Cache(c => c
        .UseQueryCache()
        .ProviderClass<HashtableCacheProvider>())
      .ShowSql())
    .Mappings(m =>                    
      m.AutoMappings.Add(AutoMap.AssemblyOf<Credit.Data.Entities.EthnicityType>(type => type.Namespace == "Credit.Data.Entities")))
    .ExposeConfiguration(BuildSchema)
    .BuildSessionFactory();
}

そしてエラー。ゆっ!

NHibernate.AssertionFailure was unhandled
  Message="null identifier"
  Source="NHibernate"
  StackTrace:
       at NHibernate.Engine.EntityKey..ctor(Object identifier, String
rootEntityName, String entityName, IType identifierType, Boolean
batchLoadable, ISessionFactoryImplementor factory, EntityMode
entityMode)
       at NHibernate.Engine.EntityKey..ctor(Object id,
IEntityPersister persister, EntityMode entityMode)
       at
NHibernate.Event.Default.AbstractSaveEventListener.PerformSaveOrReplicate
(Object entity, EntityKey key, IEntityPersister persister, Boolean
useIdentityColumn, Object anything, IEventSource source, Boolean
requiresImmediateIdAccess)
       at
NHibernate.Event.Default.AbstractSaveEventListener.PerformSave(Object
entity, Object id, IEntityPersister persister, Boolean
useIdentityColumn, Object anything, IEventSource source, Boolean
requiresImmediateIdAccess)
       at
NHibernate.Event.Default.AbstractSaveEventListener.SaveWithGeneratedId
(Object entity, String entityName, Object anything, IEventSource
source, Boolean requiresImmediateIdAccess)
       at
NHibernate.Event.Default.DefaultSaveOrUpdateEventListener.SaveWithGeneratedOrRequestedId
(SaveOrUpdateEvent event)
       at
NHibernate.Event.Default.DefaultSaveOrUpdateEventListener.EntityIsTransient
(SaveOrUpdateEvent event)
       at
NHibernate.Event.Default.DefaultSaveOrUpdateEventListener.PerformSaveOrUpdate
(SaveOrUpdateEvent event)
       at
NHibernate.Event.Default.DefaultSaveOrUpdateEventListener.OnSaveOrUpdate
(SaveOrUpdateEvent event)
       at NHibernate.Impl.SessionImpl.FireSaveOrUpdate
(SaveOrUpdateEvent event)
       at NHibernate.Impl.SessionImpl.SaveOrUpdate(Object obj)
       at FluentExample.Program.PopulateRecordTest(ISessionFactory
sessionFactory) in C:\Code\FluentExample\FluentExample\Program.cs:line
52
       at FluentExample.Program.BootstrapNH() in C:\Code\FluentExample
\FluentExample\Program.cs:line 32
       at FluentExample.Program.Main() in C:\Code\FluentExample
\FluentExample\Program.cs:line 24
       at System.AppDomain._nExecuteAssembly(Assembly assembly, String
[] args)
       at
Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
       at System.Threading.ExecutionContext.Run(ExecutionContext
executionContext, ContextCallback callback, Object state)
       at System.Threading.ThreadHelper.ThreadStart()
  InnerException:

そして、テーブル スキーマ - キックのためだけに。

USE [Credit]
GO

/****** Object:  Table [dbo].[EthnicityType]    Script Date:
08/30/2009 04:59:31 ******/
SET ANSI_NULLS ON
GO

SET QUOTED_IDENTIFIER ON
GO

CREATE TABLE [dbo].[EthnicityType](
  [ID] [uniqueidentifier] NOT NULL,
  [Title] [nvarchar](80) NOT NULL,
  [Description] [nvarchar](255) NULL,
  [IsActive] [bit] NOT NULL        
  CONSTRAINT [PK_EthnicityType_Title] PRIMARY KEY NONCLUSTERED
  (
    [ID] ASC
  )WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY
    = OFF, ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]
) ON [PRIMARY]
GO

ALTER TABLE [dbo].[EthnicityType] ADD  CONSTRAINT
  [DF_EthnicityType_ID]  DEFAULT (newid()) FOR [ID]
GO

ALTER TABLE [dbo].[EthnicityType] ADD  CONSTRAINT
  [DF_EthnicityType_IsActive]  DEFAULT ((1)) FOR [IsActive]
GO

自分の環境で自動マッピングを機能させるために多くのことを試しましたが、まだ完全には成功していません。私が試したいくつかのバリエーションは次のとおりです

  • 静的マップを使用し、Fluent Nhibernate がテーブルを再作成できるようにします。
  • Id のプライベート セットをパブリックに変更する
  • それが問題だった場合に備えて Guid を試して設定するための primarykeyconvention を作成し、これを私のマッピングに追加します。

主キーの規則:

public class PrimaryKeyConvention : IIdConvention
{
    public void Apply(IIdentityInstance instance)
    {
        instance.GeneratedBy.GuidComb();
        //instance.GeneratedBy.Native();
    }
}

アドバイスやフィードバックは大歓迎です。

4

1 に答える 1

0

Fluent NHibernate の例のエンティティはExamples.FirstProject.Entities名前空間にありますが、AutoMap をFluentExample.Entities(in type => type.Namespace == "FluentExample.Entities") に制限しています。

最初のエラー メッセージは、「保存されていない一時的なインスタンスを参照しています [...] FluentExample.Entities.Employee」を示すことで、それを示しています。

正しいインターフェイスを指すように AutoMap コードを更新する必要があります。複数のインターフェイスにクラスがある場合は、条件を変更して or を含めることができます。

于 2009-08-31T08:19:33.980 に答える