Рекурсивная инициализация гибернации
Я пишу одну функцию в hibernate, чтобы рекурсивно инициализировать все свойства объекта рекурсивно, так что загружается весь объектный объект.
У меня есть два сложных сценария, где мне нужно использовать этот
1) самокомпозитный объект, такой как категория и подкатегория...
@Entity
public class Category {
@Column(nullable = false, length = 50)
@NotNull
@Size(max = 50, message = "{50}")
protected String name;
@ManyToOne(targetEntity = Category.class, fetch = FetchType.LAZY, optional = true)
private Category parentCategory;
}
2) сложный граф объектов, у которого есть много объектов для инициализации, прежде чем он может быть использован.
Проблема заключается в том, что я не могу использовать целеустремленный выбор, потому что мне нужен весь этот график объекта только в выборочных случаях, и я хочу обобщить код, поэтому не нужно писать запросы HQL для объектов.
Я написал частичный код для этого,
public void recursiveInitliaze(Object obj) throws Exception {
if(!Hibernate.isInitialized(obj))
Hibernate.initialize(obj);
PropertyDescriptor[] properties = PropertyUtils.getPropertyDescriptors(obj);
for (PropertyDescriptor propertyDescriptor : properties) {
Object origProp = PropertyUtils.getProperty(obj, propertyDescriptor.getName());
if (origProp != null) {
this.recursiveInitliaze(origProp);
}
if (origProp instanceof Collection && origProp != null) {
for (Object item : (Collection) origProp) {
this.recursiveInitliaze(item);
}
}
}
}
но у него есть одна проблема, она заканчивается в stackoverflow для вызова метода из-за двунаправленных отношений. поэтому, как обнаружить там двунаправленную связь или есть ли лучший способ реализовать это?
Я думаю, что профиль fetch также может помочь, но все же хочет попытаться реализовать это, если это возможно, поскольку сложно настроить профиль выборки на текущем этапе проекта.
Ответы
Ответ 1
вы можете использовать Hibernate.initialize()
для объекта для его инициализации. Я не уверен, что это каскады. В противном случае вы можете проверить instanceof HibernateProxy
, который дает свойство isInitialised.
Обновление: поскольку инициализация не каскадируется, вам нужно пройти по дереву, как и вы. Используйте Hashset для отслеживания объектов, которые вы уже обработали.
Ответ 2
Полный код:
public <T> T recursiveInitliaze(T obj) {
Set<Object> dejaVu = Collections.newSetFromMap(new IdentityHashMap<Object, Boolean>());
try {
recursiveInitliaze(obj, dejaVu);
} catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) {
ReflectionUtils.handleReflectionException(e);
}
return obj;
}
private void recursiveInitliaze(Object obj, Set<Object> dejaVu) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {
if (dejaVu.contains(this)) {
return;
} else {
dejaVu.add(this);
if (!Hibernate.isInitialized(obj)) {
Hibernate.initialize(obj);
}
PropertyDescriptor[] properties = PropertyUtils.getPropertyDescriptors(obj);
for (PropertyDescriptor propertyDescriptor : properties) {
Object origProp = PropertyUtils.getProperty(obj, propertyDescriptor.getName());
if (origProp != null) {
this.recursiveInitliaze(origProp, dejaVu);
}
if (origProp instanceof Collection && origProp != null) {
for (Object item : (Collection<?>) origProp) {
this.recursiveInitliaze(item, dejaVu);
}
}
}
}
}
Код ReflectionUtils:
/**
* Handle the given reflection exception. Should only be called if no
* checked exception is expected to be thrown by the target method.
* <p>Throws the underlying RuntimeException or Error in case of an
* InvocationTargetException with such a root cause. Throws an
* IllegalStateException with an appropriate message else.
* @param ex the reflection exception to handle
*/
public static void handleReflectionException(Exception ex) {
if (ex instanceof NoSuchMethodException) {
throw new IllegalStateException("Method not found: " + ex.getMessage());
}
if (ex instanceof IllegalAccessException) {
throw new IllegalStateException("Could not access method: " + ex.getMessage());
}
if (ex instanceof InvocationTargetException) {
handleInvocationTargetException((InvocationTargetException) ex);
}
if (ex instanceof RuntimeException) {
throw (RuntimeException) ex;
}
throw new UndeclaredThrowableException(ex);
}