Как определить класс универсального типа? - PullRequest
58 голосов
/ 08 октября 2008

Я создаю универсальный класс, и в одном из методов мне нужно знать класс универсального типа, используемого в настоящее время. Причина в том, что один из методов, которые я вызываю, ожидает это в качестве аргумента.

Пример:

public class MyGenericClass<T> {
  public void doSomething() {
    // Snip...
    // Call to a 3rd party lib
    T bean = (T)someObject.create(T.class);
    // Snip...
  }
}

Очевидно, что приведенный выше пример не работает и приводит к следующей ошибке: Недопустимый литерал класса для параметра типа T.

Мой вопрос: кто-нибудь знает хорошую альтернативу или обходной путь для этого?

Ответы [ 7 ]

47 голосов
/ 08 октября 2008

Все те же проблемы: Общая информация стирается во время выполнения, ее невозможно восстановить. Обходной путь должен передать класс T в параметре статического метода:

public class MyGenericClass<T> {

    private final Class<T> clazz;

    public static <U> MyGenericClass<U> createMyGeneric(Class<U> clazz) {
        return new MyGenericClass<U>(clazz);
    }

    protected MyGenericClass(Class<T> clazz) {
        this.clazz = clazz;
    }

    public void doSomething() {
        T instance = clazz.newInstance();
    }
}

Это некрасиво, но работает.

22 голосов
/ 15 января 2011

Я только что указал на это решение:

import java.lang.reflect.ParameterizedType;

public abstract class A<B> {
    public Class<B> g() throws Exception {
        ParameterizedType superclass =
            (ParameterizedType) getClass().getGenericSuperclass();

        return (Class<B>) superclass.getActualTypeArguments()[0];
    }
}

Это работает, если A задан конкретный тип подклассом:

new A<String>() {}.g() // this will work

class B extends A<String> {}
new B().g() // this will work

class C<T> extends A<T> {}
new C<String>().g() // this will NOT work
4 голосов
/ 08 февраля 2011

К сожалению, решение Кристофа, как написано, работает только в очень ограниченных обстоятельствах. [РЕДАКТИРОВАТЬ: как прокомментировано ниже, я больше не помню свои аргументы в пользу этого предложения, и оно, вероятно, неправильно: «Обратите внимание, что это будет работать только в абстрактных классах, в первую очередь».] Следующая трудность заключается в том, что g() работает только от DIRECT подклассы A. Мы можем это исправить, хотя:

private Class<?> extractClassFromType(Type t) throws ClassCastException {
    if (t instanceof Class<?>) {
        return (Class<?>)t;
    }
    return (Class<?>)((ParameterizedType)t).getRawType();
}

public Class<B> g() throws ClassCastException {
    Class<?> superClass = getClass(); // initial value
    Type superType;
    do {
        superType = superClass.getGenericSuperclass();
        superClass = extractClassFromType(superType);
    } while (! (superClass.equals(A.class)));

    Type actualArg = ((ParameterizedType)superType).getActualTypeArguments()[0];
    return (Class<B>)extractClassFromType(actualArg);
}

Это будет работать во многих ситуациях на практике, но не ВСЕ время. Рассмотрим:

public class Foo<U,T extends Collection<?>> extends A<T> {}

(new Foo<String,List<Object>>() {}).g();

Это выдаст ClassCastException, потому что аргумент типа здесь вовсе не Class или ParameterizedType; это TypeVariable T. Так что теперь вы застряли, пытаясь выяснить, какой тип T должен был стоять и так далее по кроличьей норе.

Я думаю, что единственный разумный общий ответ - это что-то похожее на первоначальный ответ Николаса - в общем, если вашему классу нужно создавать экземпляры объектов какого-то другого класса, который неизвестен во время компиляции, пользователи вашего класса должны передавать его класс литерал (или, возможно, Фабрика) для вашего класса в явном виде и не полагаться исключительно на дженерики.

2 голосов
/ 09 ноября 2011

я нахожу другой способ получить класс универсального объекта

public Class<?> getGenericClass(){
         Class<?> result =null;
         Type type =this.getClass().getGenericSuperclass();

         if(type instanceofParameterizedType){
              ParameterizedType pt =(ParameterizedType) type;
              Type[] fieldArgTypes = pt.getActualTypeArguments();
              result =(Class<?>) fieldArgTypes[0];
        }
        return result;
  }
0 голосов
/ 29 мая 2018

открытый класс DatabaseAccessUtil {

EntityManagerFactory entitymanagerfactory;
EntityManager entitymanager;

public DatabaseAccessUtil() {
    entitymanagerfactory=Persistence.createEntityManagerFactory("bookmyshow");
    entitymanager=entitymanagerfactory.createEntityManager();
}

public void save  (T t) {
    entitymanager.getTransaction().begin();
    entitymanager.persist(t);
    entitymanager.getTransaction().commit();
}

public void update(T t) {
    entitymanager.getTransaction().begin();
    entitymanager.persist(t);
    entitymanager.getTransaction().commit();
}

public void delete(T t) {
    entitymanager.getTransaction().begin();
    entitymanager.remove(t);
    entitymanager.getTransaction().commit();
}

public Object retrieve(Query query) {
    return query.getSingleResult();
}
//call the method - retrieve(object,requiredclass.class)
public Object retrieve(Object primaryKey,class clazz) throws Exception {

    return entitymanager.find(clazz,primaryKey);    

}

}

0 голосов
/ 12 августа 2014

Я подробно остановлюсь на решении Кристофа.

Вот абстрактный класс ClassGetter:

private abstract class ClassGetter<T> {
    public final Class<T> get() {
        final ParameterizedType superclass = (ParameterizedType)
            getClass().getGenericSuperclass();
        return (Class<T>)superclass.getActualTypeArguments()[0];
    }
}

Вот статический метод, который использует вышеуказанный класс для поиска типа универсального класса:

public static <T> Class<T> getGenericClass() {
    return new ClassGetter<T>() {}.get();
}

В качестве примера его использования вы можете сделать этот метод:

public static final <T> T instantiate() {
    final Class<T> clazz = getGenericClass();
    try {
        return clazz.getConstructor((Class[])null).newInstance(null);
    } catch (Exception e) {
        return null;
    }
}

А затем используйте это так:

T var = instantiate();
0 голосов
/ 21 октября 2013

T можно легко решить с помощью TypeTools :

Class<T> t = (Class<T>) TypeResolver.resolveRawArguments(
                                MyGenericClass.class, getClass());
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...