C# Fluent NHibernate对象列表异常

C# Fluent NHibernate对象列表异常,c#,nhibernate,fluent-nhibernate,C#,Nhibernate,Fluent Nhibernate,我有一个DAL,它使用Fluent NHibernate映射查询(作为字符串存储在db中)及其参数(存储在单独的表中) 当我试图在服务层中使用这个参数列表时,我遇到了问题 List<QueryParameter> lqp = (List<QueryParameter>)qry.parameters; 投掷 初始化[nha.cs.utility.mole.Query#24]-未能延迟初始化角色集合:nha.cs.utility.mole.Query.parameters,

我有一个DAL,它使用Fluent NHibernate映射查询(作为字符串存储在db中)及其参数(存储在单独的表中)

当我试图在服务层中使用这个参数列表时,我遇到了问题

List<QueryParameter> lqp = (List<QueryParameter>)qry.parameters;
投掷

初始化[nha.cs.utility.mole.Query#24]-未能延迟初始化角色集合:nha.cs.utility.mole.Query.parameters,未关闭任何会话或会话



如果您有任何提示或建议,我们将不胜感激

谢谢


Bruce。

问题在于,您在
getQuery
中关闭会话,然后询问
Query
对象的参数列表,NHibernate试图加载该对象,但由于缺少会话而失败。要解决您的问题,您需要确保会话在整个请求过程中处于打开状态,或者至少在所有业务事务完成之前保持打开状态

我建议您采取按请求进行会话的方法。每当有web请求时打开会话,会话结束时关闭会话。这可以通过使用Global.asax文件轻松实现,您可以阅读该文件

在上面的示例中,在应用程序启动时创建会话工厂,这是推荐的,因为创建会话工厂的成本很高。然后,每个请求都会创建一个会话,并在最后进行处理。通过使用这种方法,您可以节省大量的工作,因为一切都是自动完成的

然后,要在代码中使用会话,只需调用
GetCurrentSession()
,如下所示:

var session = Global.SessionFactory.GetCurrentSession();

我希望这会有所帮助,祝你好运。我建议你考虑阅读NHiBiNATE的更好的理解。< /P>你使用的是什么版本的NHiBiNATE?你能不能也为
QueryParameter
提供你的实体/映射?我按照你的建议添加了类。实际上,这是不必要的。我不确定我在想什么。无论如何,我相信你的问题在于你的课程。调用
md.getQuery()
时是否关闭/清除会话?或者使用(会话)块将
getQuery()
方法包装到
中?如果是这样的话,那么这很可能就是你出错的原因。检查这个问题:我正在使用DAL级别的会话,在getQuery方法中打开和关闭会话。您是说我需要在使用getQuery方法的结果时保持该会话处于打开状态吗。这是一个查询对象。我添加了getQuery方法。我在这里使用会话的方式是否不正确?非常感谢您的帮助。你是对的。一旦我修改代码以在服务层创建和维护会话,问题就解决了。
List<QueryParameter> lqp = new List<QueryParameter>(qry.parameters);
IList<QueryParameter> lqp = (IList<QueryParameter>)qry.parameters;
public class Query
{
    public virtual int id { get; protected set; }
    public virtual string name { get; set; }
    public virtual string query { get; set; }
    public virtual IList<QueryParameter> parameters { get; set; }
    public virtual IList<Application> applicationsUsedIn { get; set; }

    public Query()
    {
        this.parameters = new List<QueryParameter>();
        this.applicationsUsedIn = new List<Application>();
    }

    public virtual void AddParameter(QueryParameter qp)
    {
        qp.query = this;
        this.parameters.Add(qp);
    }
}
public class QueryMap : ClassMap<Query>
{
    public QueryMap()
    {
        Table("dbo.Queries");
        Id(x => x.id);
        Map(x => x.name);
        Map(x => x.query);
        HasMany(x => x.parameters)
            .Cascade.All()
            .KeyColumn("qryid")
            .LazyLoad()
            ;
        HasManyToMany(x => x.applicationsUsedIn)
            .Table("dbo.ApplicationsQueries")
            .ParentKeyColumn("qryid")
            .ChildKeyColumn("appid")
            .Inverse()
            .LazyLoad()
            ;
    }
}
    public XmlNode runQuery(string appnname, string qryname, List<String> parms)
    {
        XmlNode xn = null;

        if ((null != appnname) && (appnname.Length > 0))
        {
            if ((null != qryname) && (qryname.Length > 0))
            {
                Query qry = md.getQuery(appnname, qryname);
                if (null != qry)
                {
                    if ((null != parms) && (parms.Count > 0)) //Passed parameters as List<String>
                    {
                        //These are the three lines I have tried
                        IList<QueryParameter> lqp = (IList<QueryParameter>)qry.parameters;
                        List<QueryParameter> lqp = qry.parameters.ToList<QueryParameter>();
                        List<QueryParameter> lqp = new List<QueryParameter>(qry.parameters);
    ...
    ...
    ...
public class QueryParameter
{
    public virtual int id { get; set; }
    public virtual Query query { get; set; }
    public virtual string name { get; set; }
    public virtual MOLEDataTypes type { get; set; }
    public virtual int order { get; set; }

    public QueryParameter()
    {
    }

    public override bool Equals(object obj)
    {
        if (obj == null)
            return false;
        var t = obj as QueryParameter;
        if (t == null)
            return false;
        if (query == t.query && name == t.name)
            return true;
        return false;
    }

    public override int GetHashCode()
    {
        return (query.id + "|" + name).GetHashCode();
    }
}
public class QueryParametersMap : ClassMap<QueryParameter>
{
    public QueryParametersMap()
    {
        Table("dbo.QueryParameters");
        Id(x => x.id);
        References(x => x.query).Column("qryid").Not.Insert();
        Map(x => x.name);
        Map(x => x.type).CustomType<MOLEDataTypes>();
        Map(x => x.order).Column("ordr");
    }
}
    public Query getQuery(string appname, string qryname)
    {
        Query retq = null;

        using (ISessionFactory isf = getSessionFactory())
        {
            using (var sess = isf.OpenSession())
            {
                using (var tran = sess.Transaction)
                {
                    try
                    {
                        tran.Begin();
                        ...
                        USING session
                        ...
                    }
                    catch (Exception ex)
                    {
                        tran.Rollback();
                        sess.Close();
                        lws.logMessage(AppName, "getQuery", ex.ToString(), MessageType.Error, MessageLevel.Error);
                    }
                }
            }
        }

        return (retq);
    }
public class Global : System.Web.HttpApplication
{    
    public static ISessionFactory SessionFactory { get; private set; }  

    protected void Application_Start(object sender, EventArgs e)
    {
        // Create a session factory when the application starts.
        // Session factories are expensive to create, and therefore you should create one and use it throughout your application.
        SessionFactory = Fluently.Configure()
                                 .Database(
                                    SQLiteConfiguration.Standard
                                        .UsingFile("firstProject.db")
                                 )
                                 .BuildSessionFactory();
    }

    protected void Application_BeginRequest(object sender, EventArgs e)
    {
        // Create a session per request.
        var session = SessionFactory.OpenSession();
        CurrentSessionContext.Bind(session);
    }

    protected void Application_EndRequest(object sender, EventArgs e)
    {
        // Close the session at the end of every request
        var session = CurrentSessionContext.Unbind(SessionFactory);
        session.Dispose();
    }

    protected void Application_End(object sender, EventArgs e)
    {
        // Close the session factory at the end of the application.
        if (SessionFactory != null)
            SessionFactory.Dispose();
    }
}
var session = Global.SessionFactory.GetCurrentSession();