2

在 Hibernate 中,我们有两个类,其中包含以下具有 JPA 映射的类:

package com.example.hibernate

import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.ManyToOne;

@Entity
public class Foo {
  private long id;
  private Bar bar;

  @Id
  @GeneratedValue(strategy = GenerationType.AUTO)
  public long getId() {
    return id;
  }

  public void setId(long id) {
    this.id = id;
  }

  @ManyToOne(fetch = FetchType.LAZY)
  public Bar getBar() {
    return bar;
  }

  public void setBar(Bar bar) {
    this.bar = bar;
  }
}

package com.example.hibernate

import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;


public class Bar {
  private long id;
  private String title;

  @Id
  @GeneratedValue(strategy = GenerationType.AUTO)
  public long getId() {
    return id;
  }

  public void setId(long id) {
    this.id = id;
  }


  public String getTitle() {
    return title;
  }

  public void setTitle(String title) {
    this.title = title;
  } 
}

现在,当我们使用会话从数据库加载类 Foo 的对象时,例如:

Foo foo = (Foo)session.get(Foo.class, 1 /* or some other id that exists in the DB*/); foo 的 Bar 成员是一个代理对象(在我们的例子中是 javassist 代理,但它可以是 cglib 之一,具体取决于您使用的字节码提供程序),它没有被初始化。如果您随后使用 session.get 获取 Bar 对象,它是刚刚加载的 Foo 类的成员(我们在同一个会话中),Hibernate 不会发出另一个 DB 查询并从会话(第一级)缓存中获取对象. 问题是这是未初始化的 Bar 类的代理,尝试调用此对象 getId() 将返回 0,而 getTitle() 将返回 null。我们当前的解决方案非常难看,它检查从 get 返回的对象是否是代理这里是代码(形成一个通用的 DAO 实现):

@SuppressWarnings("unchecked")
@Override
@Transactional(readOnly = true)
public <T extends IEntity> T get(Class<T> clazz, Serializable primaryKey) throws DataAccessException {
  T entity = (T) currentSession().get(clazz, primaryKey);
  if (entity != null) {
    if (LOG.isWarnEnabled()) {
      LOG.warn("Object not found for class " + clazz.getName() + " with primary key " + primaryKey);
    }
  } else if (entity instanceof HibernateProxy){ // TODO: force initialization due to Hibernate bug
    HibernateProxy proxy = (HibernateProxy)entity;
    if (!Hibernate.isInitialized(proxy)) {
      Hibernate.initialize(proxy);
    }
    entity = (T)proxy.getHibernateLazyInitializer().getImplementation();
  }
  return entity;
}

有没有更好的方法来做到这一点,在 Hibernate 论坛中找不到解决方案,在 Hibernate 的 JIRA 中也没有找到问题。

注意:我们不能只使用 foo.getBar() (它将正确初始化代理)来获取 Bar 类对象,因为获取 Bar 对象的 session.get 操作不知道(或不关心)Bar class 也是刚刚获取的 Foo 对象的惰性成员。

4

5 回答 5

2

我有一个类似的问题:

  • 我做了 Session.save(nastyItem) 将对象保存到会话中。但是,我没有填写映射为 update="false" insert="false" 的购房者(当您有一个组合主键时,这种情况经常发生,然后您将多对一映射为 insert="假”更新=“假”)
  • 我是一个加载项目列表的查询,而我刚刚保存的项目恰好是结果集的一部分
  • 现在出了什么问题?Hibernate 看到该项目已经在缓存中,并且 Hibernate 不会用新加载的值替换它(可能不会破坏我之前的引用 nastyItem),而是使用我自己放入 Session 缓存中的 MY nastyItem。更糟糕的是,现在买家的延迟加载被破坏了:它包含 null。

为了避免这些会话问题,我总是在保存、合并、更新或删除后进行刷新和清除。必须解决这些讨厌的问题占用了我太多时间。

于 2008-09-23T10:32:27.297 回答
0

我无法重现您所看到的行为。这是我的代码:

@Entity
public class Foo {
    private Long id; private String name; private Bar bar;

    public Foo() { }
    public Foo(String name) { this.name = name; }

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    public Long getId() { return id; }
    public void setId(Long id) { this.id = id; }

    @Basic
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }

    @ManyToOne(fetch = FetchType.LAZY)
    public Bar getBar() { return bar; }
    public void setBar(Bar bar) { this.bar = bar; }
}

@Entity
public class Bar {
    private Long id; private String name;

    public Bar() { }
    public Bar(String name) { this.name = name; }

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    public Long getId() { return id; }
    public void setId(Long id) { this.id = id; }

    @Basic
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
}

    public void testGets() {
        SessionFactory sf = new AnnotationConfiguration()
            .addPackage("hibtest")
                .addAnnotatedClass(Foo.class)
                .addAnnotatedClass(Bar.class)
            .configure().buildSessionFactory();
        Session session = null;
        Transaction txn = null;

        // Create needed data
        try {
            session = sf.openSession();
            txn = session.beginTransaction();

            // Create a Bar
            Bar bar = new Bar("Test Bar");
            session.save(bar);

            // Create a Foo
            Foo foo = new Foo("Test Foo");
            session.save(foo);

            foo.setBar(bar);

            txn.commit();
        } catch (HibernateException ex) {
            if (txn != null) txn.rollback();
            throw ex;
        } finally {
            if (session != null) session.close();
        }

        // Try the fetch
        try {
            session = sf.openSession();
            Foo foo = (Foo) session.get(Foo.class, 1L);
            Bar bar = (Bar) session.get(Bar.class, 1L);
            System.out.println(bar.getName());
        } finally {
            if (session != null) session.close();
        }
    }

正如人们所期望的那样,这一切都很好。

于 2008-09-15T07:59:34.853 回答
0

你真的需要做延迟加载吗?
您不能将 FetchType 设置为 EAGER 并始终使用连接(正确)加载它吗?

于 2008-09-15T11:55:41.540 回答
0

你做错了什么。我没有测试你的代码,但你永远不需要强制初始化代理,属性访问器会为你做这件事。如果您明确使用 Hibernate,请不要介意使用 JPA,因为您已经失去了可移植性。

Hibernate 应该在需要获取或写入数据库时​​自动检测。如果您从代理发出 getProperty(),hibernate 或任何其他 jpa 提供程序应该从数据库中获取对应的行。

我不确定 hibernate 是否足够聪明的唯一情况是,如果您发出 save() 然后发出带有已保存对象 id 的 get(),如果 save() 没有刷新可能会出现问题反对分贝。

于 2008-10-07T14:55:55.663 回答
0

没有真正看到这个问题,虽然我们确实遇到了间歇性延迟加载错误 - 所以也许我们有同样的问题,无论如何,是否可以选择使用不同的会话来加载 Bar 对象 - 应该从头开始加载它,我会期待...

于 2008-09-15T06:30:23.820 回答