1.了解Class
package com.inspire.reflection.Class_api;import java.lang.reflect.Constructor;import java.lang.reflect.Field;import java.lang.reflect.Method;public class ClassUtils { public static void main(String[] args) { String s="abc"; printclassFieldMessage(s); printClassMethodMessage(s); } /** * 打印类的信息,成员方法 * @param obj 该对象所属类的信息 */ public static void printClassMethodMessage(Object obj){ //获取类的信息,获取类的类类型 Class c1=obj.getClass();//传递的哪个子类的对象,c就是该子类的类类型 //获取类的名称 System.out.println("类名是:"+c1.getName()); //获取方法 /* Method类,方法的对象 一个成员方法就是一个Method对象 getMethods方法获取的是所有的public的方法,包括父类继承而来的 getDeclaredMethods方法获取的是所有该类自己定义的方法。不问访问权限 */ Method[] methods=c1.getMethods();// Method[] method=c1.getDeclaredMethods(); for (int i = 0; i
2.获取一个类的类对象
(1).定义一个类
public class Student { private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } private int age;}
(2).获取类对象
public class ClassDemo1 { public static void main(String[] args) { //Student的实例对象如何表示? // Student student=new Student(); /*Student也是实例对象,是Class类的实例对象。如何表示? 任何一个类都是Class类的实例对象,都有三种表示方法 */ //方式一:任何一个类都有一个隐含的静态成员class Class c1=Student.class; //方式二:已知该类的对象,通过getClass方法 Student student=new Student(); Class c2=student.getClass(); /* c1,c2表示了Student类的类类型(Class type) 万事万物皆对象 类也是对象,是Class类的实例对象 这个对象我们称为该类的类类型 */ //c1,c2都代表了Student类的类类型,一个类只可能是Class类的一个实例对象 System.out.println(c1==c2); //true //方式三: try { Class c3=Class.forName("com.inspire.reflection.Class_object.Student"); System.out.println(c2==c3); //true } catch (ClassNotFoundException e) { e.printStackTrace(); } /* 有了类的类类型,我们可以通过类类型创建该类的对象.即通过c1,c2,c3创建对象 */ try { Student student1= (Student) c1.newInstance(); //调用方法 student1.getAge(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } }}
3.通过反射调用方法
public class Printer { public void printA(){ System.out.println("无参的方法.........."); } public void printA(int a,int b){ System.out.println(a+":"+b); }}
public class MethodDemo { public static void main(String[] args) { Printer printer=new Printer(); Class c=printer.getClass(); try { //获取方法 Method method= c.getDeclaredMethod("printA", int.class, int.class);//获取自己声明的方法 //c.getMethod("printA", int.class, int.class);//获取public的方法 //方法的反射操作 method.invoke(printer,1,3); //放回方法的返回值 System.out.println("========================="); Method method1=c.getDeclaredMethod("printA"); method1.invoke(printer); } catch (Exception e) { e.printStackTrace(); } }}
4.通过反射理解泛型
java中泛型是防止错误输入的。只在编译时有效。编译之后集合的泛型是去泛型化的
public static void main(String[] args){ ArrayListlist1=new ArrayList (); list1.add("hello"); ArrayList list2=new ArrayList (); //System.out.println(c1==c2);//结果为true,说明编译之后集合的泛型是去泛型化的 Class c2=list1.getClass(); try { Method method=c2.getMethod("add", Object.class); //注意这里:虽然上面定义list1时是String类型的,但是这里可以加入int类型的数据 method.invoke(list1,100); System.out.println(list1.size()); System.out.println(list1);//不能用foreach遍历,类型不一样了 } catch (Exception e) { e.printStackTrace(); } }