2

我在使用 Hibernate 时遇到了一种奇怪的行为。我已经将我的头撞到墙上一段时间了,并且将奖励任何导致 +100 赏金解决方案的答案。

我有一个 JAX-RS (Jersey) REST 服务器,带有一个过滤器,每个请求关联一个 Hibernate 会话。

在一个请求中,客户端使用一个会话(和一个事务)发布一些存储在数据库中的数据。在随后的调用中,客户端尝试获取此实体,但 Hibernate 找不到它。

一些观察:

  • 如果我同时运行多个客户端,我只能重现这一点。我从来没有通过一次运行一个客户端来重现它。)

  • 可以在数据库中看到实体 ID,如果我重新启动服务器,那么 Hibernate 应该会找到该实体。

  • 如果我使用大小为 1 的线程池(无论我同时运行多少个客户端),则不会发生错误。

这是代码,带有一些日志记录:

chargeables.setId(new SecureRandom().nextLong());

System.out.printf("%s,  session: %s  [%s]%n",
                  Thread.currentThread(),
                  System.identityHashCode(session),
                  "session.beginTransaction()");

session.beginTransaction();


System.out.printf("%s,  session: %s  [%s]%n",
                  Thread.currentThread(),
                  System.identityHashCode(session),
                  "session.save(id = "+chargeables.getId()+")");

session.save(chargeables);


System.out.printf("%s,  session: %s  [%s]%n",
                  Thread.currentThread(),
                  System.identityHashCode(session),
                  "session.getTransaction().commit()");

session.getTransaction().commit();

获取实体的代码:

System.out.printf("%s,  session: %s  [%s]%n",
                  Thread.currentThread(),
                  System.identityHashCode(session),
                  "session.get("+id+")");

Chargeables entity = (Chargeables) session.get(Chargeables.class, id);

if (entity == null)
    System.out.printf("%s,  session: %s  [%s]%n",
                      Thread.currentThread(),
                      System.identityHashCode(session),
                      "ENTITY NOT FOUND!");

现在这里是结果日志的摘录(带有一些额外的打开/关闭会话输出):

Thread[Grizzly(5),5,main],  session:  2041842357  [factory.openSession()]
Thread[Grizzly(5),5,main],  session:  2041842357  [session.beginTransaction()]
Thread[Grizzly(5),5,main],  session:  2041842357  [session.save(id = 7939229356942262438)]
Thread[Grizzly(5),5,main],  session:  2041842357  [session.getTransaction().commit()]
Thread[Grizzly(5),5,main],  session:  2041842357  [session.close()]
[...]
Thread[Grizzly(7),5,main],  session:  1717445911  [factory.openSession()]
Thread[Grizzly(7),5,main],  session:  1717445911  [session.get(7939229356942262438)]
Thread[Grizzly(7),5,main],  session:  1717445911  [ENTITY NOT FOUND!]
Thread[Grizzly(7),5,main],  session:  1717445911  [session.close()]

我到底为什么要达到ENTITY NOT FOUND!


Hibernate 版本:4.1.9.Final
MySQL 版本:14.14 Distrib 5.5.29


映射文件Chargeables

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
       "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
       "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">

<hibernate-mapping
   default-cascade="all"
   package="se.package.common.chargeables"
   default-lazy="false">


    <class name="Chargeables">

        <id name="id" type="long">
            <generator class="assigned"/>
        </id>

        <property name="startTimeStamp" />
        <property name="endTimeStamp" />

        <list name="chargeables">
            <key column="chargeableId" />
            <list-index column="pos" />
            <many-to-many class="Chargeable"/>
        </list>

    </class>


    <class name="Chargeable">

        <id column="id" type="long">
            <generator class="native"/>
        </id>

        <discriminator />

        <property name="timestamp" />

    </class>


    <subclass name="DataTransfer" extends="Chargeable">
        <property name="bytesSent" />
        <property name="bytesReceived" />
    </subclass>


    <subclass name="TelephonyChargeable" extends="Chargeable">
        <many-to-one name="num" />
    </subclass>

    <subclass name="Call" extends="TelephonyChargeable">
        <property name="duration" />        
    </subclass>

    <subclass name="OutgoingCall" extends="Call" />
    <subclass name="IncomingCall" extends="Call" />

    <subclass name="Message" extends="TelephonyChargeable" />

    <subclass name="Sms" extends="Message" />
    <subclass name="IncomingSms" extends="Sms" />
    <subclass name="OutgoingSms" extends="Sms" />

    <subclass name="Mms" extends="Message" />
    <subclass name="IncomingMms" extends="Mms" />
    <subclass name="OutgoingMms" extends="Mms" />


</hibernate-mapping>
4

2 回答 2

0

恕我直言,这是一个隔离问题。您的第二个会话在第一个事务提交之前开始。由于默认休眠隔离级别为 read_commited,因此第二个会话无法检索实体。
你是如何在 2 个线程之间传递 id 的?

于 2013-04-04T13:08:27.973 回答
0

如果能看到数据库中的commit,则不可能是MySQL级别的隔离问题:控制台操作也是客户端操作,运行时没有特殊权限,所以符合你选择的隔离策略。

在寻找解决方案时,我发现Hibernate 提供了一些缓存数据库结果的工具:实际上那里实现了 2 级缓存。也许您的休眠安装捆绑了缓存机制?

第一级在每个会话的基础上工作,这与您在一个线程上运行代码时观察到的行为一致,并且似乎恰好在默认情况下被激活。我不精通 Hibernate,我不能肯定地说,但我认为您的设置将第一级缓存设置为只读模式,而您希望它处于读写模式。

于 2013-04-17T11:33:50.530 回答