JAVA 反射

每个类都会有一个Class对象,所有的类都是在首次使用时动态加载到JVM中。类加载器首先会检查该类的Class对象是否已经被加载,如果尚未被夹在。默认的类加载器就会根据类名查找.class文件,然后加载该类到内存。他就用来创建这个类的所有对象。

1. Class对象:

1.1 获取一个类的Class对象: Class.forName();或使用类字面常量,这样做在编译期会接受检查,不需要捕获异常。不仅普通的类,接口、数组及基本数据类型都有字面常量。类名.class

1.2  Class类。Class对象总是指向某个Class对象,可以制造类的实例,包含该类的静态成员,作用于这些事例的所有方法代码。JAVA 1.5之前使用的是普通类引用,1.5之后将类引用使用泛型语法变得更加具体。使用泛型语法的类之后,会让编译器强制执行额外的类型检查。为了使用泛化的Class引用时放松限制,可以使用通配符“?”,表示任何事物。使用普通的Class引用时若果犯错,直到运行的时候才会被发现。创建Class引用,被限定为某种类型或该类型的任何子类型,可以使用<? extends >

注:Class<?>和Class是等价的,Class<?>的好处是表示程序员就是自己选择使用非具体的版本而不是由于疏忽。

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public class TestReflection {  
  2.     public static void main(String[] args) {  
  3.         Class intClass1 = int.class;  
  4.         Class doubleClass1 = double.class;  
  5.         Class<Integer> intClass2 = int.class;// Class<Integer> intClass2 = Integer.class;  
  6.         Class<Double> doubleClass2 = double.class;  
  7.         // intClass2 = double.class; // error  
  8.           
  9.         Class<?> intClass3 = int.class;  
  10.           
  11.         Class<? extends Number> int_double_class;  
  12.         int_double_class = int.class;  
  13.         int_double_class = double.class;  
  14.     }  
  15. }  

1.3  普通的Class和泛化的Class还有个区别,在使用泛型语法的Class时使用newInstance()返回该对象的确切类型,而普通的Class返回的是Object

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. class A{  
  2.     public void print(){  
  3.         System.out.println("AAA");  
  4.     }  
  5. }  
  6. public class TestReflection {  
  7.   
  8.     public static void main(String[] args) throws InstantiationException, IllegalAccessException {  
  9.         Class aClass1 = A.class;  
  10.         Object obj = aClass1.newInstance();  
  11.         ((A)obj).print();  
  12.         Class<A> aClass2 = A.class;  
  13.         A a2 = aClass2.newInstance();  
  14.         a2.print();  
  15.     }  
  16. }  

1.4  使用泛型语法获得父类的Class引用

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. class A{}  
  2. class B extends A{}  
  3. public class TestReflection {  
  4.   
  5.     public static void main(String[] args) throws InstantiationException, IllegalAccessException{  
  6.         Class<B> bClass = B.class;  
  7.         Class<? super B> aClass1 = bClass.getSuperclass();  
  8.           
  9.         // 不能写成下面的形式,尽管在编译的时候在编译起见就知道B的父类是A  
  10.         // Class<A> aClass2 = bClass.getSuperclass(); // error  
  11.           
  12.         // 正由于上面的到的 aClass1 的含糊性,下面得到的返回值也不是精确类型而是Object  
  13.         Object aClassObj = aClass1.newInstance();  
  14.     }  
  15. }  


2. 反射相关类 : java.lang.reflect包 包括了Field、Method、Constructor类,每个类都实现了Menber接口。这些类对象都是在JVM运行时创建的。

反射机制提供了足够的支持,使得能够创建一个在编译时完全未知的对象,并调用此对象的方法。

2.1 获取类的构造函数、属性和方法

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. class A{  
  2.     private int i;  
  3.     public A (int x) {  
  4.         i = x;  
  5.     }  
  6.     public void print() {  
  7.         System.out.println("AAA");  
  8.     }  
  9.       
  10. }  
  11. public class TestReflection {  
  12.     public static void main(String[] args) throws InstantiationException, IllegalAccessException{  
  13.         Class<?> aClass = null;  
  14.         try {  
  15.             aClass = Class.forName("reflection.A");  
  16.         } catch (ClassNotFoundException e) {  
  17.             e.printStackTrace();  
  18.         }  
  19.         System.out.println("------------类所有方法------------");  
  20.         Method [] methods = aClass.getDeclaredMethods();  
  21.           
  22.         for (Method method : methods) {  
  23.             System.out.println(method.toString());  
  24.         }  
  25.         System.out.println("------------类构造方法------------");  
  26.         Constructor [] constructors = aClass.getDeclaredConstructors();  
  27.         for (Constructor constructor : constructors) {  
  28.             System.out.println(constructor.toString());  
  29.         }  
  30.         System.out.println("------------类属性------------");  
  31. //      Field [] fields = aClass.getFields();//获取public的  
  32.         Field [] fields = aClass.getDeclaredFields();  //获取public的  
  33.         for (Field field : fields) {  
  34.             System.out.println(field.toString());  
  35.         }  
  36.     }  
  37. }  
  38. //getFields()与getDeclaredFields()区别:  
  39. //getFields()只能访问类中声明为公有的字段,getDeclaredFields()能访问类中所有的字段  
  40. //getMethods()与getDeclaredMethods()区别:  
  41. //getMethods()只能访问类中声明为公有的方法,能访问从其它类继承来的公有方法,getDeclaredFields()能访问类中所有的字段,不能访问从其它类继承来的方法  
  42. //getConstructors()只能访问类中声明为public的构造函数,getDeclaredConstructors()能访问类中所有的构造函数  

2.2 创建对象

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. class A{  
  2.     private int i;  
  3.     public A(){}  
  4.     private A (int x) {  
  5.         i = x;  
  6.     }  
  7.     public void print() {  
  8.         System.out.println("AAA");  
  9.     }  
  10.     public int getI() {  
  11.         return i;  
  12.     }  
  13.     public void setI(int i) {  
  14.         this.i = i;  
  15.     }  
  16. }  
  17. public class TestReflection {  
  18.     public static void main(String[] args){  
  19.         try {  
  20.             Class<A> aClass = A.class;  
  21.             //创建对象的两种方式  
  22.             // 1. Class.newInstance  
  23.              aClass.newInstance();// 必须有默认的构造函数  
  24.             // 2. Constructor.newInstance  
  25.             Constructor<A> intConstructors = aClass.getDeclaredConstructor(int.class); // new Class<?> []{int.class}  
  26.             intConstructors.setAccessible(true);  
  27.             A a = intConstructors.newInstance(100);  
  28.             System.out.println(a.getI());  
  29.         } catch (Exception e) {  
  30.             // TODO Auto-generated catch block  
  31.             e.printStackTrace();  
  32.         }  
  33.     }  
  34. }  

2.3 访问类属性 一般用来修改源码中类成员的值

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. class A{  
  2.     private static boolean isOpen = false;  
  3.     public static boolean isOpen() {  
  4.         return isOpen;  
  5.     }  
  6. }  
  7. public class TestReflection {  
  8.     public static void main(String[] args){  
  9.         try {  
  10.             System.out.println(A.isOpen());  
  11.             Field openField = A.class.getDeclaredField("isOpen");  
  12.             openField.setAccessible(true);  
  13.             openField.setBoolean(new A(), true);  
  14.             System.out.println(A.isOpen());  
  15.         } catch (Exception e) {  
  16.             e.printStackTrace();  
  17.         }  
  18.     }  
  19. }  
 

2.4 调用函数

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. class A{  
  2.     private void print() {  
  3.         System.out.println("AAA");  
  4.     }  
  5. }  
  6. public class TestReflection {  
  7.     public static void main(String[] args){  
  8.         try {  
  9.             A a = new A();  
  10.             Class<A> aClass = (Class<A>) a.getClass();  
  11.             aClass.getDeclaredMethod("print");  
  12.             Method printMethod = aClass.getDeclaredMethod("print");//Method printMethod = aClass.getDeclaredMethod("print", new Class<?>[]{});  
  13.             printMethod.setAccessible(true);  
  14.             printMethod.invoke(a);  
  15.         } catch (Exception e) {  
  16.             e.printStackTrace();  
  17.         }  
  18.     }  
  19. }  
 

注:利用反射调用私有属性、构造函数或者方法时要设置Accessible属性为true。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值