Spring Boot 和 Hibernate 懒加载的奇怪行为

3
我希望您能协助翻译编程相关内容,从Play Framework转向Spring Boot时遇到了一些难以理解的问题。
考虑以下简单实体:
@Entity
@Table(name = "system")
public class System {

  @Id
  @Column(name = "systemid", unique = true, nullable = false, length = 36)
  public String systemid;

  @ManyToOne(fetch = FetchType.LAZY, optional=false)
  @JoinColumn(name = "systemtypeid", nullable = false)
  public Systemtype systemtype;

  //This column is added just for testing purposes, see comments below
  @Column(name = "systemtypeid", insertable=false, updatable=false)
  public String systemtypeid;

  @OneToMany(fetch = FetchType.LAZY, mappedBy = "system")
  public Set<SystemItem> items = new HashSet<SystemItem>(0);
}


@Entity
@Table(name = "systemtype")
public class Systemtype {

  @Id
  @Column(name = "systemtypeid", unique = true, nullable = false, length = 36)
  @Access(AccessType.PROPERTY)
  public String systemtypeid;

  public String getSystemtypeid() {
    return systemtypeid;
  }

  public void setSystemtypeid(String systemtypeid) {
    this.systemtypeid = systemtypeid;
  }

  @Column(name = "name", length = 60)
  public String name;

  @OneToMany(fetch = FetchType.LAZY, mappedBy = "systemtype")
  public Set<System> systems = new HashSet<System>(0);
  }

@Entity
@Table(name = "systemitem")
public class SystemItem {

  @Id
  @Column(name = "systemitemid", unique = true, nullable = false, length = 36)
  public String systemitemid;

  @Column(name = "name", length = 60)
  public String name;

  @ManyToOne(fetch = FetchType.LAZY)
  @JoinColumn(name = "systemid", nullable = false)
  public System system;
}

控制器:

    @RestController
    @RequestMapping(value = ApiController.SYSTEM_URL)
    public class SystemController extends ApiController {

      private static final Logger log = LoggerFactory.getLogger(SystemController.class);

      @Autowired
      private SystemService systemService;

      @RequestMapping(method = RequestMethod.GET)
      public Collection<System> getSystems() throws Exception {
        List<System> systems = systemService.getSystems();
        return systems;
      }
}

最后,服务部分:

@Service
@Transactional(readOnly = true, value = "tmPrimary")
public class SystemService {
  private static final Logger log = LoggerFactory.getLogger(SystemService.class);

  @Autowired
  SystemRepository systemRepository; //Spring-Data's PagingAndSortingRepository

  public List<System> getSystems() {
    return Lists.newArrayList(systemRepository.findAll());
  }
}

当在控制器中调用getSystems()方法时,我希望得到一个填充了基本字段的所有系统列表,因为其他所有内容都是懒加载的。这也确实发生了,我还检查了Hibernate查询,唯一被查询的表确实是System表。到目前为止一切正常。
但我的第一个问题是,我也希望system.systemtype.systemtypeid被填充,因为这是系统表中的外键,但这始终为空。如果我使用EclipseLink,我相信这将是预期的行为,但在Hibernate中不应该是这样的。 我添加了一个System对象的虚拟列(systemtypeid)以进行验证,这确实得到了填充。所以: system.systemtypeid = "Something" system.systemtype.systemtypeid = null
我认为这两个都应该是"Something",那么我错过了什么吗?
我在此期间找到了解决方法,只有在注释使用属性级别访问时,Hibernate 才会获取外键 ID。这里有说明: https://developer.jboss.org/wiki/HibernateFAQ-TipsAndTricks#jive_content_id_How_can_I_retrieve_the_identifier_of_an_associated_object_without_fetching_the_association 当使用 Play 时,我没有遇到这种行为,我现在猜测 Play 可能会在幕后执行一些操作来生成它们。
第二个问题开始于将数据序列化为 JSON 以便发送到客户端。
首先我希望能够得到惰性初始化异常,因为转换是在控制器内完成的,而事务注释方法在服务上。令人惊讶的是,这并没有发生,项目集被惰性加载,并且所有项都可以很好地序列化为 JSON。 再次无法理解这种行为,为什么惰性加载在控制器内部进行时仍然有效?Spring Boot 是否在控制器方法开始时打开会话?
我已经理解了这个问题(感谢Nico指导),Spring Boot默认启用OpenSessionInView(非常糟糕的想法),因此我必须在我的applications.properties中添加spring.jpa.open-in-view=false。
更令人惊讶的是,即使这样,system.systemtype仍将为空,也不会进行懒加载。唯一能够加载system.systemtype的方法是将其声明为急切加载。
也许我错过了一些明显的东西,但我很难理解这种行为,这与我使用Play时所经历的完全不同,我认为这种行为应该是完全相同的。
在所有编辑之后,唯一剩下的问题是为什么system.systemtype永远不会进行懒加载。我进行了一些测试,只有当我为Systemtype实体的所有字段添加getter/setter时,它才会进行懒加载。这正常吗?
查看Hibernate文档,现在似乎是这样: https://docs.jboss.org/hibernate/orm/3.3/reference/en/html/performance.html "默认情况下,Hibernate3对于集合使用延迟选择获取,对于单值关联使用延迟代理获取。这些默认设置对于大多数应用程序中的大多数关联关系是有意义的。"
"代理获取:当调用与关联对象相关联的除标识符getter之外的方法时,将获取单值关联。"
这是否意味着所有用于单值关联的实体都需要定义其所有字段的getter和setter才能进行延迟加载?这听起来非常令人困惑,因为它在所有其他场景中的工作方式都不同。
在Play中,我从未需要使用getter/setter,所以也许Play改变了单值关联的默认获取策略?甚至有没有一种方法可以做到这一点?"

你将 systemType 设置为懒加载 fetch = FetchType.LAZY,所以它是 null。你能发布一下你的 Hibernate 配置吗? - Rafik BELDI
你尝试过使用私有属性和getter/setter吗? 我不知道Hibernate如何管理公共属性和延迟加载。 - Eria
Hibernate 不需要 getters/setters。就像你所看到的,它可以正常获取所有 systemItems。 - mfc
2个回答

0
如果你将fetch type设置为懒加载,你会得到一个空对象,直到你试图访问它,然后它就被填充了。这是在Spring中的行为,我不确定其他平台是否也一样。
如果你想保持外键实体仍然是懒加载,并且你仍然想单独访问外键id,那么你必须像这样向你的Entity添加一个虚拟属性:
@Column(name = "system_type_id", insertable = false, updatable = false)
public Integer systemTypeId;

关于序列化延迟加载的对象,说实话我不是很确定,但我猜想当Jackson尝试序列化该对象时,它会调用getter方法并相应地填充延迟加载的对象。我不明白为什么序列化后延迟加载的对象仍然为空,这没有道理,如果数据已经被获取那么对象应该被填充了。

这是Hibernate或所使用的JPA实现的行为,与Spring无关。 - dunni
我非常了解延迟加载机制,这不是我的问题所在。我认为你没有完全阅读我的帖子,因为你建议我添加的那个字段已经存在了。无论如何,Hibernate 不应该要求这个额外的字段,除非期间发生了一些变化,因为我正在使用 Play 和 Hibernate,外键总是被填充到 systemtype 对象中。然而,EclipseLink 是需要的。 - mfc
好的,很抱歉我无法提供更多帮助 :) 希望有更有经验的人能尽快解决这个问题。如果你在这里看到http://stackoverflow.com/questions/24526216/how-can-i-fetch-foreign-key-id-directly-instead-of-entire-entity-in-case-of-lazy,这种情况已经持续了一年。如果它能像你描述的那样工作就太好了,可以通过lazyobject.id访问延迟对象的id。祝好运。 - prettyvoid
我编辑了我的第一篇帖子,因为我解决了SystemType的问题,但是第二个问题仍然让我困扰。 - mfc

0

不,我没有使用它。除非这是由Spring Boot在底层添加的? - mfc
1
似乎是这样:https://dev59.com/0Izda4cB1Zd3GeqPn4_9 我认为Spring Boot默认启用此功能是一个非常糟糕的想法。 - mfc

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接