English 中文(简体)
Can I do this Generic thing?
原标题:

It seems I m missing something with Java Generics because something I think is simple, it appears to me that can´t be done. Maybe you can help...

This is the scenario: I m coding a generic abstract DAO with simple CRUD operation so every specific DAO of my application can have it for free:

public abstract DefaultDAO<T,V> {
  private EntityManager manager;

  public BaseDAO(EntityManager em) {
    this.manager = em;
  }

  public void create(T entity) {
    manager.persist(entity);
  }

  // You know the others...

  public T read(V pk) {
    // Now, here is the problem. 
    // EntityManager signature is: public <T> T find(Class<T> entityClass, Object primaryKey);
    // So I must provide the type of the object this method will be returning and 
    // the primary key.
    // resulting object will be T typed and pk type is V type (although is not needed to type it since the method expects an Object)
    // So... I expected to be allowed to do something like this
    return manager.find(T, pk); // But it s not allowed to use T here. T is not an instance
  }   
}

Now I would go and implement an specific DAO:

public PersonDAO extends DefaultDAO<PersonEntity, Long> {
  public PersonDAO(EntityManager em) {
    super(em);
  }
  // CRUD methods are inherited
}

And client code for my DAO would be:

EntityManager manager = ...
PersonDAO dao = new PersonDAO(manager);
Long pk = .....
PersonEntity person = dao.find(pk); // DAO would return a PersonEntity

When client executes code, BaseDAO knows the type of entity it must return and the type of the primary key of that entity because I set it on the specific dao, but I don t know how to code the read() method correctly.

Hope you can help. Thanks a lot!

最佳回答
问题回答

You re trying to use a type parameter as if it s a normal expression. You can t do this. In other languages you may be able to get the class for the type parameter at execution time, but you can t in Java due to type erasure.

In this case you ll need to pass in the Class<T> at execution time - for example to the constructor:

public abstract class DefaultDAO<T, V> {

    private EntityManager manager;
    private final Class<T> clazz;

    public DefaultDAO(EntityManager em, Class<T> clazz) {
        this.manager = em;
        this.clazz = clazz;
    }

    public T read(V pk) {
        return manager.find(clazz, pk);
    }
}

Java doesn t have the generic class information at runtime anymore (it is called Type Erasure). What I did was to give my Abstract Daos the instance of the generic class.

public abstract DefaultDAO<T,V> {

  protected Class<T> genericClass;
  private EntityManager manager;

  protected BaseDAO(EntityManager em, Class<T> implclass) {
    this.manager = em;
  }

  public void create(T entity) {
    manager.persist(entity);
  }

  // You know the others...

  public T read(V pk) {
    return manager.find(this.getGenericClass(), pk); // But it s not allowed to use T here. T is not an instance
  }

  public Class<T> getGenericClass()
  {
    return genericClass;
  }
}

public class Blubb
{
    private String id;

    // Getters and Stuff...
}

public class BlubbDao extends DefaultDAO<Blubb, String>
{
    public BlubbDao(EntityManager em)
    {
        super(em, Blubb.class);
    }
}

I can t promise that it will run out of the box but I hope you get the idea.

There is a way to do this using reflection, as long as your classes follow a consistent class hierarchy in terms of generics (i.e. any intermediate classes in the inheritance hierarchy between your base class and your concrete class use the same generic parameters in the same order).

We use something like this in our abstract class, defined as HibernateDAO where T is the entity type and K is the PK:

private Class getBeanClass() {
    Type daoType = getClass().getGenericSuperclass();
    Type[] params = ((ParameterizedType) daoType).getActualTypeArguments();
    return (Class) params[0];
}

which smells a bit but trades off the ickiness of passing a .class up from the concrete implementation in a constructor for the ickiness of insisting that you keep your type hierarchy consistent in that way.

I think you can do that:

 public <T> T find(Class<T> clazz, Object id) {
        return entityManager.find(clazz, id);
    }




相关问题
Spring Properties File

Hi have this j2ee web application developed using spring framework. I have a problem with rendering mnessages in nihongo characters from the properties file. I tried converting the file to ascii using ...

Logging a global ID in multiple components

I have a system which contains multiple applications connected together using JMS and Spring Integration. Messages get sent along a chain of applications. [App A] -> [App B] -> [App C] We set a ...

Java Library Size

If I m given two Java Libraries in Jar format, 1 having no bells and whistles, and the other having lots of them that will mostly go unused.... my question is: How will the larger, mostly unused ...

How to get the Array Class for a given Class in Java?

I have a Class variable that holds a certain type and I need to get a variable that holds the corresponding array class. The best I could come up with is this: Class arrayOfFooClass = java.lang....

SQLite , Derby vs file system

I m working on a Java desktop application that reads and writes from/to different files. I think a better solution would be to replace the file system by a SQLite database. How hard is it to migrate ...

热门标签