5

NHibernateISessionの Castle によって生成されたを閉じる必要がありますか? ISessionManagerそれらの とのトランザクションを処理するにはどうすればよいISessionですか? 私はまだNHibernateに慣れていません。

編集:遅延読み込みを行いたいのですが、次のメッセージが表示されます:

初期化中 [ロールのコレクションを遅延初期化できませんでした: 、セッションまたはセッションが閉じられませんでした」

これは、特定のインスタンスを実装するために継承する汎用リポジトリです。

[Transactional]
public class Repository<TKey, TModel>
    : IRepository<TKey, TModel>
    where TKey : IComparable
    where TModel : class
{
    private readonly ISessionManager _sessionManager;

    protected ISession Session { get { return _sessionManager.OpenSession(); } }

    public Repository(ISessionManager sessionManager)
    {
        _sessionManager = sessionManager;
    }
    #region IRepository<TKey,TModel> Members

    public virtual TModel Select(TKey key)
    {
        using (var session = _sessionManager.OpenSession())
        {
            return session.Get<TModel>(key);
        }
    }

    public virtual IList<TModel> SelectWhere(Func<TModel, bool> query)
    {
        using (var session = Session)
        {
            return session.Linq<TModel>().Where(query).ToList();
        }
    }

    public virtual TModel Single(Func<TModel, bool> query)
    {
        using (var session = Session)
        {
            return session.Linq<TModel>().SingleOrDefault(query);
        }
    }

    public virtual TModel First(Func<TModel, bool> query)
    {
        using (var session = Session)
        {
            return session.Linq<TModel>().FirstOrDefault(query);
        }
    }

    public virtual IList<TModel> All()
    {
        using (var session = Session)
        {
            return session.Linq<TModel>().ToList();
        }
    }

    [Transaction(TransactionMode.Requires)]
    public virtual void Store(TModel entity)
    {
        using (var session = Session)
        {
            session.SaveOrUpdate(entity);
        }
    }

    [Transaction(TransactionMode.Requires)]
    public virtual void Store(IEnumerable<TModel> entities)
    {
        using (var session = Session)
        {
            foreach (TModel entity in entities)
                session.SaveOrUpdate(entity);
        }
    }


    [Transaction(TransactionMode.Requires)]
    public virtual void Remove(TModel entity)
    {
        using (var session = Session)
        {
            session.Delete(entity);
        }

    }

    public virtual void Remove(Func<TModel, bool> query)
    {
        IEnumerable<TModel> entities = SelectWhere(query);
        Remove(entities);
    }

    [Transaction(TransactionMode.Requires)]
    public virtual void Remove(IEnumerable<TModel> entities)
    {
        using (var session = Session)
        {
            foreach (TModel entity in entities)
                session.Delete(entity);
        }
    }

    #endregion
}

public class Repository<TModel>
    : Repository<Guid, TModel>, IRepository<TModel>
    where TModel : class
{
    public Repository(ISessionManager sessionManager) : base(sessionManager) { }
}

public class Repository
    : Repository<ulong, object>, IRepository
{
    public Repository(ISessionManager sessionManager) : base(sessionManager) { }
}

そのリポジトリを呼び出すサンプルを次に示します。

IUserRepository userRepository = new UserRepository(); // This is actually provided by my IoC

var users = userRepository.All();
foreach (var user in Users)
{
    foreach (var picture in user.Pictures)
    {
        // I get exceptions when I do stuff like this.
    }
}
4

2 に答える 2

6

はい、必ず廃棄してISessionください。使用法に関するドキュメントをISessionManager参照してください。

トランザクションについては、自動トランザクション機能の使用を検討してください。

SessionManagerは ATM に対応しているため、ISession必要に応じて をスマートに破棄し、明らかにISession.

これは、ASP.NET MVC + Castle Automatic Transaction Facility + NHibernate 機能を使用するクイック & ダーティ サンプル アプリです。

于 2009-07-13T13:19:48.410 に答える
0

using ステートメントを含むトランザクションを使用して、破棄を処理します。

public void Save<K>(K entity)
        {

            if (entity == null)
                throw new ArgumentNullException("item", "The item being saved cannot be null.");

            using (ISession session = GetSession())
            {
                using (ITransaction tx = session.BeginTransaction())
                {
                    session.SaveOrUpdate(entity);
                    tx.Commit();
                }
            }
        }

同じアクションで変更を行った後にオブジェクトにアクセスすると、遅延読み込みエラーが発生します。保存後にオブジェクトにアクセスしないことでエラーを修正しました。ここに説明があります: NHibernate.LazyInitializationException

階層を正しく保存していないことが原因だと思います。私はそれをテストしていませんが、親オブジェクトにアクセスしたい場合は、親オブジェクトを保存すると問題が解決する可能性があります。保存後にアクセスする必要がある情報を、保存前にローカル変数に入れるだけで問題が解決したようです。

于 2009-07-16T14:14:59.420 に答える