NHibernate要求.NET将字符串参数转换为整数,不知道为什么

时间:2010-07-13 17:06:27

标签: c# .net nhibernate fluent-nhibernate

(编辑以避免导致错误的道路)

在提供详细信息之前,非常简短的版本:我有一个我需要退出NHibernate的SQL语句,并且我有Criteria API语句给我。 NHibernate试图对它生成的语句做错了,我正试图找出原因。

我在Fluent NHibernate中有以下映射:

public class StaffClass {
    public virtual int Staff_DBID { get; private set; }
    public virtual int Class_DBID { get; private set; }

    // Equals and GetHasCode overrides not shown
    // other code originally here removed as it was distracting from the point and didn't change anything
}

public class StaffClassMap : ClassMap<StaffClass> {
    public StaffClassMap() {
        CompositeId()
            .KeyProperty(x => x.Staff_DBID)
            .KeyProperty(x => x.Class_DBID);
    }
}

对于名为ClassStudent的表,存在一组类似的代码。这两个表都是连接表。

我有以下针对域对象运行的代码:

            using (transaction = session.BeginTransaction()) {

                var criteria = session.CreateCriteria(typeof(Student));

                var staffClasses = DetachedCriteria.For<StaffClass>()
                    .Add(Restrictions.Eq("Staff_DBID", desiredStaffDBID))
                    .SetProjection(Projections.Property("Class_DBID"));

                var studentClasses = DetachedCriteria.For<ClassStudent>()
                    .Add(Subqueries.In("Class_DBID", staffClasses))
                    .SetProjection(Projections.Property("Student_DBID"));

                criteria.Add(Subqueries.In("Student_DBID", studentClasses));

                var students = criteria.List<Student>();

                foreach (var student in students) {
                    Console.WriteLine(string.Format("Student: {0}, {1}", student.LastName, student.FirstName));
                }
            }

当我尝试运行此代码时,我收到以下异常:

NHibernate.ADOException occurred
  Message=could not execute query
[ SELECT this_.Student_DBID as Student1_7_0_, this_.DistrictStudentID as District2_7_0_, this_.LastName as LastName7_0_, this_.FirstName as FirstName7_0_, this_.MidName as MidName7_0_, this_.School_DBID as School6_7_0_ FROM [Student] this_ WHERE @p0 in (SELECT this_0_.Student_DBID as y0_ FROM [ClassStudent] this_0_ WHERE @p1 in (SELECT this_0_0_.Class_DBID as y0_ FROM [StaffClass] this_0_0_ WHERE this_0_0_.Staff_DBID = @p2)) ]
Positional parameters:  #0>Student_DBID #1>Class_DBID #2>3664
[SQL: SELECT this_.Student_DBID as Student1_7_0_, this_.DistrictStudentID as District2_7_0_, this_.LastName as LastName7_0_, this_.FirstName as FirstName7_0_, this_.MidName as MidName7_0_, this_.School_DBID as School6_7_0_ FROM [Student] this_ WHERE @p0 in (SELECT this_0_.Student_DBID as y0_ FROM [ClassStudent] this_0_ WHERE @p1 in (SELECT this_0_0_.Class_DBID as y0_ FROM [StaffClass] this_0_0_ WHERE this_0_0_.Staff_DBID = @p2))]
  Source=NHibernate
  SqlString=SELECT this_.Student_DBID as Student1_7_0_, this_.DistrictStudentID as District2_7_0_, this_.LastName as LastName7_0_, this_.FirstName as FirstName7_0_, this_.MidName as MidName7_0_, this_.School_DBID as School6_7_0_ FROM [Student] this_ WHERE @p0 in (SELECT this_0_.Student_DBID as y0_ FROM [ClassStudent] this_0_ WHERE @p1 in (SELECT this_0_0_.Class_DBID as y0_ FROM [StaffClass] this_0_0_ WHERE this_0_0_.Staff_DBID = @p2))
  StackTrace:
       at NHibernate.Loader.Loader.DoList(ISessionImplementor session, QueryParameters queryParameters)
       at NHibernate.Loader.Loader.ListIgnoreQueryCache(ISessionImplementor session, QueryParameters queryParameters)
       at NHibernate.Loader.Loader.List(ISessionImplementor session, QueryParameters queryParameters, ISet`1 querySpaces, IType[] resultTypes)
       at NHibernate.Loader.Criteria.CriteriaLoader.List(ISessionImplementor session)
       at NHibernate.Impl.SessionImpl.List(CriteriaImpl criteria, IList results)
       at NHibernate.Impl.CriteriaImpl.List(IList results)
       at NHibernate.Impl.CriteriaImpl.List[T]()
       at Test.Program.Main(String[] args) in C:\Projects\Test\Test\Program.cs:line 86
  InnerException: System.FormatException
       Message=Failed to convert parameter value from a String to a Int32.
       Source=System.Data
       StackTrace:
            at System.Data.SqlClient.SqlParameter.CoerceValue(Object value, MetaType destinationType)
            at System.Data.SqlClient.SqlParameter.GetCoercedValue()
            at System.Data.SqlClient.SqlParameter.Validate(Int32 index, Boolean isCommandProc)
            at System.Data.SqlClient.SqlCommand.BuildParamList(TdsParser parser, SqlParameterCollection parameters)
            at System.Data.SqlClient.SqlCommand.BuildExecuteSql(CommandBehavior behavior, String commandText, SqlParameterCollection parameters, _SqlRPC& rpc)
            at System.Data.SqlClient.SqlCommand.RunExecuteReaderTds(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, Boolean async)
            at System.Data.SqlClient.SqlCommand.RunExecuteReader(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, String method, DbAsyncResult result)
            at System.Data.SqlClient.SqlCommand.RunExecuteReader(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, String method)
            at System.Data.SqlClient.SqlCommand.ExecuteReader(CommandBehavior behavior, String method)
            at System.Data.SqlClient.SqlCommand.ExecuteDbDataReader(CommandBehavior behavior)
            at System.Data.Common.DbCommand.System.Data.IDbCommand.ExecuteReader()
            at NHibernate.AdoNet.AbstractBatcher.ExecuteReader(IDbCommand cmd)
            at NHibernate.Loader.Loader.GetResultSet(IDbCommand st, Boolean autoDiscoverTypes, Boolean callable, RowSelection selection, ISessionImplementor session)
            at NHibernate.Loader.Loader.DoQuery(ISessionImplementor session, QueryParameters queryParameters, Boolean returnProxies)
            at NHibernate.Loader.Loader.DoQueryAndInitializeNonLazyCollections(ISessionImplementor session, QueryParameters queryParameters, Boolean returnProxies)
            at NHibernate.Loader.Loader.DoList(ISessionImplementor session, QueryParameters queryParameters)
       InnerException: System.FormatException
            Message=Input string was not in a correct format.
            Source=mscorlib
            StackTrace:
                 at System.Number.StringToNumber(String str, NumberStyles options, NumberBuffer& number, NumberFormatInfo info, Boolean parseDecimal)
                 at System.Number.ParseInt32(String s, NumberStyles style, NumberFormatInfo info)
                 at System.String.System.IConvertible.ToInt32(IFormatProvider provider)
                 at System.Convert.ChangeType(Object value, Type conversionType, IFormatProvider provider)
                 at System.Data.SqlClient.SqlParameter.CoerceValue(Object value, MetaType destinationType)

现在让我烦恼的是,生成的SQL正是我希望SQL看起来的样子。如果我在SQL Server Managmeent Studio(SQL 2008)中使用手动参数替换运行SQL,我会得到一个正确的结果集:

SELECT
    this_.Student_DBID as Student1_7_0_, 
    this_.DistrictStudentID as District2_7_0_, 
    this_.LastName as LastName7_0_, 
    this_.FirstName as FirstName7_0_, 
    this_.MidName as MidName7_0_, 
    this_.School_DBID as School6_7_0_ 
FROM
    [Student] this_ 
WHERE 
    Student_DBID in (
        SELECT 
            this_0_.Student_DBID as y0_ 
        FROM
            [ClassStudent] this_0_ 
        WHERE Class_DBID in (
            SELECT
                this_0_0_.Class_DBID as y0_ 
            FROM
                [StaffClass] this_0_0_ 
            WHERE
                this_0_0_.Staff_DBID = 3664
        )
    )

尝试的参数转换来自哪里?为什么会这样?我该怎么做才能避免这个问题?

谢谢!

2 个答案:

答案 0 :(得分:1)

首先,您要将ID列映射两次:作为参考,并作为复合ID的一部分。

这是错误的,你应该使用流利的等效<key-many-to-one/>映射id成员。

但是还有更多...如果这个类没有任何数据,为什么要映射呢?

如果在课堂上有一组职员会更好,反之亦然(映射为多对多)。

我假设您的查询是针对“员工”的学生。

型号:

public class Student : Entity<Guid>
{
}

public class Staff : Entity<Guid>
{
}

public class Class : Entity<Guid>
{
    public virtual ICollection<Student> Students { get; set; }
    public virtual ICollection<Staff> Staff { get; set; }
}

这些集合上的映射只是多对多的,所以我会把它留下来。

使用HQL比使用Criteria更容易进行查询:

var students = session.CreateQuery("see the query below")
                      .SetParameter("staff", id)
                      .List<Student>();

from Student s
where s in
        (select elements(c.Students)
         from Class c
         where :staff in elements(c.Staff))

答案 1 :(得分:0)

只是关闭这个问题 - 我放弃说服NHibernate做我想做的事情。我决定使用General SQL Parser和真正的命令和连接提供程序的包装器。 http://blog.theagileworkshop.com/2009/06/09/using-schemas-with-sqlite-for-in-memory-nhibernate-tests/描述了由于不同原因的类似任务。在任何情况下,这都避免了整个问题,因为它迫使我在SQL级别工作,这只能让我做我想要的,而不会让NHibernate妨碍,但也允许HQL甚至直接的DB访问被截获。