JAVA反射突破泛型的约束

        JDK1.5增加泛型之后,集合操作也应用了泛型约束,比如ArrayList<String>,则该对象放进去的内容对象都是String类型的,这是将同一类对象放在一起。如果再放其他类型的对象进去,编译器就报错了。如下例子。

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

public class BreakGenericConstraint {

	public static void main(String[] args) throws Exception{
		List<Integer> list=new ArrayList<Integer>();
		list.add(1);
		list.add(2);
		list.add("abc");
	    System.out.println(list);
	}
}
     如果是在eclipse下编译,文件有红色x。DOS下编译,如下图报错



          那如果我就是想把'abc'放进去,是否可以呢?可以,用反射。

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

public class BreakGenericConstraint {

	public static void main(String[] args) throws Exception{
		List<Integer> list=new ArrayList<Integer>();
		list.add(1);
		list.add(2);
	    Method m=ArrayList.class.getDeclaredMethod("add", Object.class);
	    m.invoke(list, "abc");
	    System.out.println(list);
	}
}

运行结果

[1, 2, abc]
[Ljava.lang.reflect.Type;@3ec86a


     用反编译工具查看class文件,得到的源码如下

import java.io.PrintStream;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

public class BreakGenericConstraint
{
  public static void main(String[] args)
    throws Exception
  {
    List list = new ArrayList();
    list.add(Integer.valueOf(1));
    list.add(Integer.valueOf(2));
    Method m = ArrayList.class.getDeclaredMethod("add", new Class[] { Object.class });
    m.invoke(list, new Object[] { "abc" });
    System.out.println(list);
  }
}

      也就是说,泛型的约束只在编译器,由编译器进行检查约束,所以使用在运行期使用反射,就可以跳过编译器的拦截了。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 17中,可以使用反射来获取泛型类型。面是一种常见的方法: 1. 首先,通过反射获取目标类的Class对象。假设目标类为`MyClass`,可以使用`MyClass.class`或者`Class.forName("com.example.MyClass")`来获取。 2. 使用`getDeclaredField`方法获取目标字段的Field对象。假设目标字段为`myField`,可以使用`Class.getDeclaredField("myField")`来获取。 3. 通过Field对象的`getGenericType`方法获取字段的泛型类型。这将返回一个Type对象,表示字段的实际类型。 4. 如果字段的类型是参数化类型(即包含泛型参数),可以通过Type对象的一些方法来获取泛型参数的信息。例如,可以使用`ParameterizedType`接口来获取参数化类型的原始类型泛型参数列表。 下面是一个示例代码,演示了如何使用反射获取泛型类型: ```java import java.lang.reflect.Field; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; public class Main { public static void main(String[] args) throws NoSuchFieldException { // 获取目标类的Class对象 Class<MyClass> clazz = MyClass.class; // 获取目标字段的Field对象 Field field = clazz.getDeclaredField("myField"); // 获取字段的泛型类型 Type fieldType = field.getGenericType(); // 如果字段的类型是参数化类型 if (fieldType instanceof ParameterizedType) { ParameterizedType parameterizedType = (ParameterizedType) fieldType; // 获取参数化类型的原始类型 Type rawType = parameterizedType.getRawType(); System.out.println("Raw type: " + rawType); // 获取参数化类型泛型参数列表 Type[] typeArguments = parameterizedType.getActualTypeArguments(); for (Type typeArgument : typeArguments) { System.out.println("Type argument: " + typeArgument); } } } } class MyClass { private List<String> myField; } ``` 在上面的示例中,我们通过反射获取了`MyClass`类中名为`myField`的字段的泛型类型。输出结果如下: ``` Raw type: interface java.util.List Type argument: class java.lang.String ``` 这表明`myField`字段的类型是`List<String>`,其中`List`是原始类型,`String`是泛型参数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值