NHibernate QuerySyntaxException

时间:2009-08-04 15:22:54

标签: nhibernate exception

我跟随Summer of NHibernate Screencast Series并且遇到了一个奇怪的NHibernate异常。

NHibernate.Hql.Ast.ANTLR.QuerySyntaxException:
Exception of type
'Antlr.Runtime.NoViableAltException' was thrown.
[select from DataTransfer.Person p where p.FirstName=:fn].

我通过以下方式偏离了Screencast系列:

  1. 针对MS SQL Server Compact数据库运行
  2. 我使用的是MSTest而不是MbUnit
  3. 我已尝试过任意数量的查询组合,但结果总是相同。我目前的CreateQuery语法

    public IList<Person> GetPersonsByFirstName(string firstName)
    {
        ISession session = GetSession();
    
        return session.CreateQuery("select from Person p " +
            "where p.FirstName=:fn").SetString("fn", firstName)
            .List<Person>();
    }
    

    虽然不是直接查询,但此方法有效

    public Person GetPersonById(int personId)
    {
        ISession session = GetSession();
        return session.Get<Person>(personId);
    }
    

    我的 hibernate.cfg.xml

    <?xml version="1.0" encoding="utf-8" ?>
    <hibernate-configuration xmlns="urn:nhibernate-configuration-2.2">
      <session-factory name="BookDb">
        <property name="connection.provider">NHibernate.Connection.DriverConnectionProvider</property>
        <property name="connection.driver_class">NHibernate.Driver.SqlServerCeDriver</property>
        <property name="dialect">NHibernate.Dialect.MsSqlCeDialect</property>
        <property name="connection.connection_string">Data Source=C:\Code\BookCollection\DataAccessLayer\BookCollectionDb.sdf</property>
        <property name="show_sql">true</property>
        <property name="proxyfactory.factory_class">NHibernate.ByteCode.Castle.ProxyFactoryFactory, NHibernate.ByteCode.Castle</property>
        <mapping assembly="DataTransfer"/>
      </session-factory>
    </hibernate-configuration>
    

    Person.hbm.xml

    <?xml version="1.0" encoding="utf-8" ?>
    <hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" assembly="DataTransfer" namespace="DataTransfer">
      <class name="DataTransfer.Person,DataTransfer" table="Person">
        <id name="PersonId" column="PersonId" type="Int32" unsaved-value="0">
          <generator class="native"/>
        </id>
        <property name="FirstName" column="FirstName" type="String" length="50" not-null="false" />
        <property name="LastName" column="LastName" type="String" length="50" not-null="false" />
      </class>
    </hibernate-mapping>
    

2 个答案:

答案 0 :(得分:18)

我也跟随Summer of NHibernate Screencast Series并遇到了同样的问题。

问题在于HQL“从用户p中选择”将其更改为“从用户p 中选择p”或只是“从用户p ”。

  

在NHibernate版本1.2下的截屏视频中使用的'速记'HQL表单在2.0中被弃用,在2.1.x中被消除,因为默认查询解析器已切换成为更严格的选择。

public IList<Person> GetPersonsByFirstName(string firstName)
{
    ISession session = GetSession();

    return session.CreateQuery("select p from Person p where p.FirstName=:fn")
                              .SetString("fn", firstName)
                              .List<Person>();
}

答案 1 :(得分:5)

由于您在<hibernate-mapping元素中指定了命名空间,因此可以编写:

 <class name="Person" table="Person">
  ....

尝试之后,如果它不起作用 - 我不知道它为什么不起作用。我已经尝试了很多你给出的例子并且它有效。

我已经看到新的解析器会抛出一些奇怪的错误,你只需要通过反复试验来解决它:(。

修改

关于试验和错误:您可以将查询更改为“来自人员”,看看是否有效(如果没有...我被卡住了)。然后添加过滤器,首先直接尝试p.FirstName ='x'。然后尝试使用参数。您可以尝试不添加别名。

另外,请尝试使用最新版本的NH。

修改2

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" assembly="NHibernateTests, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" namespace="NHibernateTests">

<class name="User" table="`User`" xmlns="urn:nhibernate-mapping-2.2">
<id name="Id" type="Int32" column="UserId">
  <generator class="assigned" />
</id>
<property name="UserName" type="String">
  <column name="UserName" not-null="true" />
</property>
<property name="FName" type="String">
  <column name="FName" />
</property>
  </class></hibernate-mapping>

和查询:

IList<User> users = session.CreateQuery("select from User p " +
                              "where p.UserName=:fn").SetString("fn", "u")
                  .List<User>();

像魅力一样工作。