NHibernate - LAZY LOADING PRBLEM -Initializing [] - Не удалось инициализировать прокси - нет сеанса. "} - PullRequest
7 голосов
/ 28 сентября 2011

Привет, я использую Fluent NHibernate, и я немного путаюсь с отложенной загрузкой.

  1. Я запросил объект в базе данных
  2. измененные свойства объекта
  3. обновите базу данных с помощью этогоobject

Вот код:

public class Credentials
{
    public virtual int Id { get; set; }
    public virtual string Nick { get; set; }
    public virtual string Password { get; set; }
}

public class CredentialsMap : ClassMap<Credentials>
{
    public CredentialsMap()
    {
        Id(x => x.Id);
        Map(x => x.Nick).Column("NICK");
        Map(x => x.Password).Column("PASSWORD");
        Table("R_CREDENTAILS");
    }
}

public class Status
{
    public virtual int Id { get; set; }
    public virtual string Msg { get; set; }
    public virtual DateTime AddTime { get; set; }
}

public class StatusMap : ClassMap<Status>
{
    public StatusMap()
    {
        Id(x => x.Id);
        Map(x => x.Msg).Column("MESSAGE");
        Map(x => x.AddTime).Column("ADD_TIME");
        Table("R_STATUS");
    }
}

public class Account
{
    public virtual int Id { get; set; }
    public virtual string SelfNick { get; set; }
    public virtual Credentials Credentials { get; set; }
    public virtual Status Status { get; set; }
}

public class AccountMap : ClassMap<Account>
{
    public AccountMap()
    {
        Id(x => x.Id);
        Map(x => x.SelfNick).Column("SELF_NICK");
        References(x => x.Credentials)
            .Column("CREDENTIALS_ID")
            .ForeignKey();
        References(x => x.Status)
            .Column("STATUS_ID")
            .ForeignKey();
        Table("R_ACCOUNTS");
    }
}

Класс конфигурации NHibernate:

public class NHiberanteHelper
{
    private static ISessionFactory _sessionFactory;

    private static ISessionFactory SessionFactory
    {
        get
        {
            if (_sessionFactory == null)
                InitializeSessionFactory();

            return _sessionFactory;
        }
    }

    private static void InitializeSessionFactory()
    {
        _sessionFactory = Fluently.Configure()
            .Database(MsSqlConfiguration.MsSql2008
                          .ConnectionString(
                               @"Server=TEST\SQLEXPRESS;Database=SimpleNHibernate;Trusted_Connection=True;").
                               ShowSql()
                              )
            .Mappings(m => m.FluentMappings.AddFromAssemblyOf<Account>().Conventions.Add( DefaultCascade.All()))
            .ExposeConfiguration(cfg => new SchemaUpdate(cfg).Execute(true, true))
            .BuildSessionFactory();
    }

    public static ISession OpenSession()
    {
        return SessionFactory.OpenSession();
    }
}

Вот использование:

    public class LoginDbHelper
    {
        public static Account GetAccount(string nick)
        {
            using (var session = NHiberanteHelper.OpenSession())
            {
                var account = (session.QueryOver<Account>()
                    .JoinQueryOver<Credentials>(a => a.Credentials)
                    .Where(c => c.Nick == nick));

                if (account != null)
                    return account.SingleOrDefault();

                return null;
            }
        }

        public static void SaveOrUpdateAccount(Account account)
        {
            using (var session = NHiberanteHelper.OpenSession())
            {
                using (var trans = session.BeginTransaction())
                {
                    session.SaveOrUpdate(account);
                    trans.Commit();
                }
            }
        }
   }

Проблемакод:

var actualAccount = LoginDbHelper.GetAccount(nick);

//modify
actualAccount.Status.Msg = "New status 2";
actualAccount.Status.AddTime = DateTime.Now;


LoginDbHelper.SaveOrUpdateAccount(account);

Я получаю эту ошибку:

{"Initializing[NHibernateSample1.Status#1]-Could not initialize proxy - no Session."}

StackTrace:

 at NHibernate.Proxy.AbstractLazyInitializer.Initialize() in d:\CSharp\NH\NH\nhibernate\src\NHibernate\Proxy\AbstractLazyInitializer.cs:line 113
   at NHibernate.Proxy.AbstractLazyInitializer.GetImplementation() in d:\CSharp\NH\NH\nhibernate\src\NHibernate\Proxy\AbstractLazyInitializer.cs:line 191
   at NHibernate.ByteCode.Castle.LazyInitializer.Intercept(IInvocation invocation) in d:\CSharp\NH\NH\nhibernate\src\NHibernate.ByteCode.Castle\LazyInitializer.cs:line 61
   at Castle.DynamicProxy.AbstractInvocation.Proceed()
   at Castle.Proxies.StatusProxy.set_Msg(String value)
   at NHibernateSample1.Program.Main(String[] args) in E:\C# PROJECTS\Samples\SimpleNHibernateClient\NHibernateSample1\Program.cs:line 215
   at System.AppDomain._nExecuteAssembly(RuntimeAssembly assembly, String[] args)
   at System.AppDomain.ExecuteAssembly(String assemblyFile, Evidence assemblySecurity, String[] args)
   at Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()

Я гуглю и думаю, что это вызвано Lazy Loadingпотому что в методе GetAccount я закрываю SESSION.Это моя первая попытка использования NHibernate, поэтому КАК МОЖНО ПРАВИЛЬНО РЕШИТЬ ЭТУ ПРОБЛЕМУ?Можно отключить LAZY LOADING, если ДА, как это сделать?

Ответы [ 3 ]

8 голосов
/ 28 сентября 2011

Вы правы.Поскольку сеанс NHibernate закрыт в вашем методе GetAccount (он открыт только в области действия оператора using), вы не можете загружать дополнительные объекты вне этого метода.Существует 2 возможных исправления:

  1. Создайте сеанс на уровне операции (т. Е. В методе, содержащем код проблемы), затем используйте этот сеанс в методах get & save.Вы можете использовать сеанс, передавая его в качестве параметра методам.
  2. Измените объект, чтобы он не использовал отложенную загрузку.Вы можете сделать это, добавив .Not.LazyLoad() к Status объекту в вашем беглом отображении.
4 голосов
/ 29 сентября 2011

Я считаю, что самый простой способ отключить отложенную загрузку - это добавить соглашение DefaultLazy, т.е.

.Conventions.Add( DefaultCascade.All(), DefaultLazy.Never() )

Обратите внимание, что включение отложенной загрузки (DefaultLazy.Always ()) действительно может повысить производительность в зависимости от вашего приложения.

Недостатком является то, что вам всегда нужно открывать сеанс, прежде чем вы сможете лениво загрузить остальные данные в объект. По моему опыту, управление сессиями для поддержки отложенной загрузки - одна из самых больших проблем с NHibernate.

0 голосов
/ 28 сентября 2011

Вы открываете и закрываете сеанс в методе LoginDbHelper.GetAccount (...).
Попробуйте создать и открыть сеанс вне метода и передайте его как параметр метода, например:

    // method
    public static Account GetAccount(string nick, ISession session) 
    {   
     var account = (session.QueryOver<Account>().JoinQueryOver<Credentials>(a => a.Credentials).Where(c => c.Nick == nick));   

           if (account != null)                      
            return account.SingleOrDefault();                    
    return null;  
    }

// usage
    var actualAccount = LoginDbHelper.GetAccount(nick);   
    actualAccount.Status.AddTime = DateTime.Now;   
    using (var session = NHiberanteHelper.OpenSession())  

    LoginDbHelper.SaveOrUpdateAccount(account, session); 
...