Question

I have a Hibernate class which is essentially just a wrapper around loads of collections.

So the class is (massively simplified/pseudo) something like:

@Entity  
public class MyClass {

  @OneToMany  
  Map1

  @OneToMany  
  Map2

  @OneToMany   
  Map3

  AddToMap1();  
  AddToMap2();  
  AddToMap3();  
  RemoveFromMap1();  
  RemoveFromMap2();  
  RemoveFromMap3();  
  DoWhateverWithMap1();  
  DoWhateverWithMap2();  
  DoWhateverWithMap3();  

}

etc. Each of those Maps then has a few methods associated with it (add/remove/interrogate/etc).

As you can imagine, by the time I added the 10th collection or so, the class is getting a tad ridiculous in size.

What I'd love to do is something along the lines of:

 
@Entity  
public class MyClass {

  ClassWrappingMap1;

  ClassWrappingMap2;

  ClassWrappingMap3;
}

With all the various methods wrapped up in those classes:

public class ClassWrappingMap1 {

  @OneToMany
  Map

  AddToMap();  
  RemoveFromMap();  
  DoWhateverWithMap();  

}

I thought perhaps I could use @Embedded for this, but I don't seem to be able to get it to work (Hibernate simply doesn't even try to persist the Map inside the wrapperClass).

Has anyone ever done something like this before? Any hints?

Many thanks,
Ned

Was it helpful?

Solution

Hibernate manual for annotations states following:

While not supported by the EJB3 specification, Hibernate Annotations allows you to use association annotations in an embeddable object (ie @*ToOne nor @*ToMany). To override the association columns you can use @AssociationOverride.

So your wrapping approach should work.


First of all, you should check all log files etc for any related errors.

You could try something like this:

  • In your master class (MyClass )
@Entity  
public class MyClass {

  @Embedded
  ClassWrappingMap1 map1;
}
  • And in your wrapping class
@Embeddable
public class ClassWrappingMap1 {

  @OneToMany
  Map map1;

}

Notice that ClassWrappingMap1 uses @Embeddable annotation. However, according to docs the @Embeddable annotation should not be needed, it should be default when @Embedded annotation is used.

Make sure that every ClassWrappingMap class maps a different column in database. Also ClassWrappingMap classes should not have a primary key (@Id or @EmbeddedId columns).

OTHER TIPS

Although i do not know a default strategy when using a wrapper class, you could use a Hibernate Interceptor to initialize your wrapper's by overriding onLoad method. Something like

public class WrapperInterceptor extends EmptyInterceptor {

    private Session session;

    public boolean onLoad(Object entity, Serializable id, Object[] state, String[] propertyNames, Type[] types) {
        if (entity instanceof MyClass) {
             MyClass myClass = (MyClass) entity;

             Query query = session.createQuery(<QUERY_TO_RETRIEVE_WRAPPED_ENTITY_GOES_HERE>);

             WrappedEntity wrappedEntity = query.list().get(0);

             myClass.setWrapperClass(new WrapperClass(wrappedEntity));
        }
    }

    public void setSession(Session session) {
        this.session = session;
    }
}

Takes care of the following:

A client using this interceptor must set Session property

So your code looks like this one

WrapperInterceptor interceptor = new WrapperInterceptor();

Session session = sessionFactory().openSession(interceptor);

Transaction tx = session.beginTransaction();

interceptor.setSession(session);

MyClass myClass = (MyClass) session.get(newItem, myClassId); // Triggers onLoad event

tx.commit();
session.close();

Or use Spring AOP to do the same task. See Domain Driven Design with Spring and Hibernate

If you know another strategy, share it with us.

regards,

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top