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);
}
}
也就是说,泛型的约束只在编译器,由编译器进行检查约束,所以使用在运行期使用反射,就可以跳过编译器的拦截了。