在Java / Hibernate应用程序中,我有两个类Cat
和Kitten
的双向关系,如下图所示:
public class Cat {
...
@OneToMany(mappedBy="cat", fetch = FetchType.LAZY)
@OnDelete(action = OnDeleteAction.CASCADE)
@LazyCollection(LazyCollectionOption.EXTRA)
@Getter
@Setter
private List<Kitten> kittens = new LinkedList();
public void addKitten(Kitten k) {
kittens.add(k);
}
...
}
public class Kitten {
...
@ManyToOne(fetch=FetchType.LAZY)
@Getter
@Setter
private Cat cat;
...
}
在一个巨大的for循环中,20000 Kitten
被添加到之前创建的不同Cat
实体中。 for循环中的重要代码如下所示:
....
Kitten k = new Kitten();
k.setAttribut("foo");
k.setCat(currentCat); // (a) line
currentCat.addKitten(k); // (b) line
daoFactory.getKittenDao().save(k);
...
代码正在运行,但不知何故,性能非常慢。在之前的迭代(单向)中,应用程序要快得多。由于最终版本应该在1000000 Kitten
上运行,因此必须有一种改进的方法。如果我对上面代码的时间进行基准测试,则大约需要40秒。如果我通过删除线(a)或(b)来模拟单向,则在两种情况下都需要10秒(但是如果我访问对象则会稍后出现)。
所以我的问题是: 我是否想念一些东西,或者Hibernate中双向关系的内部维护是否很慢?由于模拟单向更快,我预计双向运行时间约为15秒。
更新
保存实体的代码位于SAX-Parser DefaultHandler实现中。因此,在解析XML结构时,首先Cat
保存在startElement()
函数中,稍后Kitten
将保存在另一个startElement()
调用中。
关于@Bartun的建议/问题:我看了一下批处理。问题是,通过使用DAO和startElement()
函数,很难说,何时保存了50个实体来刷新会话。计数器可以做到这一点。但是,它没有解释通过建立双向性对性能的影响。
由于事务管理Spring @Transactional
用于启动XML解析的函数。
答案 0 :(得分:1)
我已将处理时间缩短到我可以忍受的值。它并没有真正解决我的问题,但显着减少了时间。如果其他人有同样的问题,这里是当前的代码和一个简短的解释。
public class Cat {
...
@OneToMany(mappedBy="cat", fetch = FetchType.LAZY)
@OnDelete(action = OnDeleteAction.CASCADE)
@LazyCollection(LazyCollectionOption.EXTRA)
@Getter
@Setter
private Set<Kitten> kittens = new HashSet();
public void addKitten(Kitten k) {
k.setCat(this);
if (Hibernate.isInitialized(kittens)) kittens.add(k); //line X
}
...
}
public class Kitten {
...
@ManyToOne(fetch=FetchType.LAZY)
@OnDelete(action = OnDeleteAction.CASCADE)
@Getter
@Setter
private Cat cat;
...
}
line X
。在原始代码中,每次添加一只小猫时,都会从数据库中加载整个小猫列表。当我找到here时,如果列表尚未被初始化,则不得添加小猫。我知道,这必须小心,因为每个小猫必须在首次初始化列表之前保存在数据库中。否则一切都会失去同步。List
到Set
的变化也是一个很小的改进,以便稍后在代码中启用多个fetch joins
。 答案 1 :(得分:0)
你在哪里开业?确保您使用的是一笔交易
此外,你应该使用批量查询这样的金额
有关批量https://vladmihalcea.com/how-to-batch-insert-and-update-statements-with-hibernate/
的详细信息您还可以使用hibernate statictics检查刷新计数和其他事项 https://www.thoughts-on-java.org/how-to-activate-hibernate-statistics-to-analyze-performance-issues/
当您处理如此数量的内容时,您需要完全控制刷新/批处理