为了账号安全,请及时绑定邮箱和手机立即绑定

Java 反射:怎么取出类的泛型类型

Java 反射:怎么取出类的泛型类型

慕妹3242003 2019-03-14 15:11:08
[code="java"]public class BaseHello {private Class entityClass; public BaseHello(){      //entityClass怎么赋值?(怎么能知道entityClass就是代表Personal这个类?)  }}[/code][code="java"]public static void main(String[] args) {BaseHello pdao = new BaseHello();System.out.println(pdao);}[/code]如注释的问题:在BaseHello的构造方法中,怎么给entityClass赋值?
查看完整描述

7 回答

?
森栏

TA贡献1810条经验 获得超5个赞

ParameterizedType parameterizedType = (ParameterizedType)this.getClass().getGenericSuperclass();

entityClass= (Class)(parameterizedType.getActualTypeArguments()[0]);


查看完整回答
反对 回复 2019-04-26
?
BIG阳

TA贡献1859条经验 获得超6个赞

可以看下这个,这是rapid-framework通用框架,里面有讲Base,

[code="java"]public class UserDao extends BaseHibernateDao{

@Override  
public Class getEntityClass() {  
    return User.class;  
}

}[/code]

你看下这样获取是否是你要的


查看完整回答
反对 回复 2019-04-26
?
烙印99

TA贡献1829条经验 获得超13个赞

不要再纠结这个问题了,这个没有解决方案,曾经和你一样的想法,记不清哪儿看的了,这个牵涉到泛型的擦除远离。

大意是:
1. 泛型编译时会推导擦除
2. 只有在编译时保存了类型信息时才能获得,运行时的泛型信息因为类型擦除所以肯定获取不到泛型的具体信息

详细测试见:
[code="java"]
public class TypesTest extends TestCase {
Map a;
Inner b;
List[][]> c;
List d;
Set e;
Outer.Inner f;

    private ParameterizedType mapStringInteger;
    private ParameterizedType innerFloatDouble;
    private ParameterizedType listSetStringArray;
    private ParameterizedType listString;
    private ParameterizedType setString;
    private ParameterizedType outerInner;
    private GenericArrayType setStringArray;

    @Override
    protected void setUp() throws Exception {
            super.setUp();
            mapStringInteger = (ParameterizedType) getClass().getDeclaredField("a")
                            .getGenericType();
            innerFloatDouble = (ParameterizedType) getClass().getDeclaredField("b")
                            .getGenericType();
            listSetStringArray = (ParameterizedType) getClass().getDeclaredField(
                            "c").getGenericType();
            listString = (ParameterizedType) getClass().getDeclaredField("d")
                            .getGenericType();
            setString = (ParameterizedType) getClass().getDeclaredField("e")
                            .getGenericType();
            outerInner = (ParameterizedType) getClass().getDeclaredField("f")
                            .getGenericType();
            setStringArray = (GenericArrayType) listSetStringArray
                            .getActualTypeArguments()[0];

            System.out.println(UtilJson.getObjectMapper()
                            .writerWithType(TypesTest.class).writeValueAsString(this));
            ;
    }

    class Inner<T1, T2> {
    }

    static class Outer<T> {
            class Inner {
            }
    }

}
[/code]


查看完整回答
反对 回复 2019-04-26
?
动漫人物

TA贡献1815条经验 获得超10个赞

因为类的成员变量在编译时会保留其类型信息的。
如下E是泛型类型,Set e,在类型推导时会替换为Object的,所以你无法获取到类型信息,而e2,定义时就包含了类型信息,故可以获取到。
[code="java"]
public class TypesTest {
Set e;
Set e2;
}
[/code]

完整测试代码如下
[code="java"]
import java.lang.reflect.GenericArrayType;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.junit.Test;

public class TypesTest {
Map a;
Inner b;
List[][]> c;
List d;
Set e;
Outer.Inner f;

private ParameterizedType mapStringInteger;
private ParameterizedType innerFloatDouble;
private ParameterizedType listSetStringArray;
private ParameterizedType listString;
private ParameterizedType setString;
private ParameterizedType outerInner;
private GenericArrayType setStringArray;

private String toString(ParameterizedType parameterizedType) {
    Type[] types = parameterizedType.getActualTypeArguments();
    String ret = "\n\t" + parameterizedType + "\n\t\t泛型个数:" + types.length + "==>";
    for (Type type : types) {
        ret += type + ", ";
    }
    return ret;
}

@Test
public void main() throws Exception {
    mapStringInteger = (ParameterizedType) getClass().getDeclaredField("a").getGenericType();
    innerFloatDouble = (ParameterizedType) getClass().getDeclaredField("b").getGenericType();
    listSetStringArray = (ParameterizedType) getClass().getDeclaredField("c").getGenericType();
    listString = (ParameterizedType) getClass().getDeclaredField("d").getGenericType();
    setString = (ParameterizedType) getClass().getDeclaredField("e").getGenericType();
    outerInner = (ParameterizedType) getClass().getDeclaredField("f").getGenericType();
    setStringArray = (GenericArrayType) listSetStringArray.getActualTypeArguments()[0];

    System.out.println("a Map<String, Integer> 推导擦除后类型信息:" + toString(mapStringInteger));
    System.out.println();
    System.out.println("b Inner<Float, Double> 推导擦除后类型信息:" + toString(innerFloatDouble));
    System.out.println();
    System.out.println("c List<Set<String>[][]> 推导擦除后类型信息:" + toString(listSetStringArray));
    System.out.println();
    System.out.println("d List<String> 推导擦除后类型信息:" + toString(listString));
    System.out.println();
    System.out.println("e Set<String> 推导擦除后类型信息:" + toString(setString));
    System.out.println();
    System.out.println("f Outer<String>.Inner 推导擦除后类型信息:" + toString(outerInner));
    System.out.println();
    System.out.println("c List<Set<String>[][]> List第二层Set的泛型推导信息:" + setStringArray);
}

class Inner<T1, T2> {
}

static class Outer<T> {
    class Inner {
    }
}

}


查看完整回答
反对 回复 2019-04-26
?
桃花长相依

TA贡献1860条经验 获得超8个赞

你调用的无参的构造函数:
BaseHello pdao = new BaseHello();

相当于变成了这样:
[code="java"]
public class BaseHello {

private Personal entityClass;

 public BaseHello(){   
     //entityClass怎么赋值?(怎么能知道entityClass就是代Personal这个类?) 
       你觉得这个地方你能怎么赋值?参数都不带!你想给entityClass赋值,就需要用带参数的构造函数。简单的事情不要整复杂了。
 }  

  public BaseHello(Class<T> arg){   
  这个地方你晓得怎么赋值了吧?

}

}


查看完整回答
反对 回复 2019-04-26
?
眼眸繁星

TA贡献1873条经验 获得超9个赞

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;

/**

  • 反射工具类
    */
    @SuppressWarnings("unchecked")
    public class ReflectUtils {

    /**

    /**

    public PojoRawMapper() {
    entityClass = ReflectUtils.getClassGenricType(getClass());
    }

    • 根据索引获得超类的参数类型

    • @param clazz 超类类型

    • @param index 索引 */ @SuppressWarnings("rawtypes") public static Class getClassGenricType(final Class clazz, final int index) { Type genType = clazz.getGenericSuperclass(); if (!(genType instanceof ParameterizedType)) { return Object.class; } Type[] params = ((ParameterizedType)genType).getActualTypeArguments(); if (index >= params.length || index < 0) { return Object.class; } if (!(params[index] instanceof Class)) { return Object.class; } return (Class) params[index]; } } [/code] [code="java"] public class PojoRawMapper implements RowMapper{ protected Class entityClass;

    • 获得超类的参数类型,取第一个参数类型

    • @param 类型参数

    • @param clazz 超类类型 */ @SuppressWarnings("rawtypes") public static Class getClassGenricType(final Class clazz) { return getClassGenricType(clazz, 0); }


查看完整回答
反对 回复 2019-04-26
  • 7 回答
  • 0 关注
  • 1463 浏览

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号