zoukankan      html  css  js  c++  java
  • Java反射基本玩法

    三个主要的反射类


      Class反射对象描述类语义结构,可以从Class对象中获取构造函数、成员变量、方法类等元素的反射对象,并以编程的方式通过这些反射对象对目标类对象进行操作。这些反射对象类在java.reflect包中定义,下面是最主要的三个反射类:

    • Constructor:类的构造函数反射类,通过Class#getConstructors()方法可以获得类的所有构造函数反射对象数组。在JDK5.0中,还可以通过getConstructor(Class... parameterTypes)获取拥有特定入参的构造函数反射对象。Constructor的一个主要方法是newInstance(Object[] initargs),通过该方法可以创建一个对象类的实例,相当于new关键字。在JDK5.0中该方法演化为更为灵活的形式:newInstance(Object... initargs)。
    • Method:类方法的反射类,通过Class#getDeclaredMethods()方法可以获取类的所有方法反射类对象数组Method[]。在JDK5.0中可以通过getDeclaredMethod(String name, Class... parameterTypes)获取特定签名的方法,name为方法名;Class...为方法入参类型列表。Method最主要的方法是invoke(Object obj, Object[] args),obj表示操作的目标对象;args为方法入参,下面代码清单中3.处演示了这个反射类的使用方法。在JDK5.0中,该方法的形式调整为invoke(Object obj, Object... args)。此外,Method还有很多用于获取类方法更多信息的方法:
      1. Class getReturnType():获取方法的返回值类型;
      2. Class[] getParameterTypes():获取方法的入参类型数组;
      3. Class[] getExceptionTypes():获取方法的异常类型数组;
      4. Annotation[][] getParameterAnnotations():获取方法的注解信息,JDK5.0中的新方法;
    • Field:类的成员变量的反射类,通过Class#getDeclaredFields()方法可以获取类的成员变量反射对象数组,通过Class#getDeclaredField(String name)则可获取某个特定名称的成员变量反射对象。Field类最主要的方法是set(Object obj, Object value),obj表示操作的目标对象,通过value为目标对象的成员变量设置值。如果成员变量为基础类型,用户可以使用Field类中提供的带类型名的值设置方法,如setBoolean(Object obj, boolean value)、setInt(Object obj, int value)等。
    • 除此之外,Java还为包提供了Package反射类,在JDK5.0中还为注解提供了AnnotatedElement反射类。
     1 /**
     2  * Copyright 2016 Zhengbin's Studio.
     3  * All right reserved.
     4  * 2016年7月1日 下午2:24:19
     5  */
     6 package Nine;
     7 
     8 import java.lang.reflect.Constructor;
     9 import java.lang.reflect.Field;
    10 import java.lang.reflect.InvocationTargetException;
    11 import java.lang.reflect.Method;
    12 
    13 /**
    14  * @author zhengbinMac
    15  *
    16  */
    17 public class ReflectTest {
    18 
    19     public static Car initByDefaultConst() throws ClassNotFoundException,... {
    20         // 1.通过类装载器获取Car类对象
    21         ClassLoader loader = Thread.currentThread().getContextClassLoader();
    22 //        Class clazz = loader.loadClass("Nine.Car");
    23         Class clazz = Class.forName("Nine.Car");
    24         // 2.获取类的默认构造器对象并通过它实例化Car
    25         Constructor cons = clazz.getDeclaredConstructor((Class[])null);
    26         Car car = (Car)cons.newInstance();
    27         
    28         car.introduce();
    29         
    30         // 3.通过反射方法设置属性
    31         Method setBrand = clazz.getMethod("setBrand", String.class);
    32         setBrand.invoke(car, "BMW");
    33         Method setColor = clazz.getMethod("setColor", String.class);
    34         setColor.invoke(car, "red");
    35         Method setMaxSpeed = clazz.getMethod("setMaxSpeed", int.class);
    36         setMaxSpeed.invoke(car, 280);
    37         
    38         // 4.通过反射方法设置类的成员变量
    39         Field fild = clazz.getDeclaredField("NUM");
    40         fild.set(car, 2);
    41         
    42 //        car.setBrand("BMW");
    43 //        car.setColor("red");
    44 //        car.setMaxSpeed(280);
    45         return car;
    46     }
    47     
    48     public static void main(String[] args) throws ClassNotFoundException,... {
    49         Car car = initByDefaultConst();
    50         car.introduce();
    51         System.out.println(car.getNUM());
    52     }
    53 }

      总之,Java的反射体系保证了可以通过程序化的方式访问目标类中所有的元素,对于private或protected的成员变量和方法,只要JVM的安全机制允许,也可以通过反射进行调用。

    对于private或protected的成员变量的和方法的反射调用


      通过下面代码说明这个问题:

    1 /**
    2  * @author zhengbinMac
    3  */
    4 public class PrivateCar {
    5     private String color;
    6     protected void drive() {
    7         System.out.println("drive private car! the color is:" + color);
    8     }
    9 }
     1 /**
     2  * @author zhengbinMac
     3  */
     4 public class PrivateCarReflect {
     5     public static void main(String[] args) throws ClassNotFoundException,... {
     6         Class clazz = Class.forName("Nine.PrivateCar");
     7         PrivateCar car = (PrivateCar)clazz.newInstance();
     8         Field field = clazz.getDeclaredField("color");
     9         // 1.取消Java语言访问检查以访问private变量
    10         field.setAccessible(true);
    11         field.set(car, "红色");
    12         Method driveMtd = clazz.getDeclaredMethod("drive",(Class[])null);
    13         // 2.取消Java语言访问检查以访问protected方法
    14         driveMtd.setAccessible(true);
    15         driveMtd.invoke(car, (Object[])null);
    16     }
    17 }

      将输出:drive private car! the color is:红色

      在访问private、protected成员变量和方法时必须通过setAccessible(boolean access)方法取消Java语言检查,否则将抛出IllegalAccessException。如果JVM的安全管理器设置了相应的安全机制,调用该方法将抛出SecurityException。

    另一篇反射博客链接

  • 相关阅读:
    ASP.NET购物车Cookie获取,创建,添加,更新,删除的用法
    python-列表常用功能介绍
    Java并发编程(一)
    《How Tomcat Works》读书笔记(三)--Connector(连接器)
    好了歌
    《How Tomcat Works》读书笔记(二)
    《Head First Servlet&JSP》读书笔记
    J2EE思考
    《Spring实战》读书笔记--使用SpringMVC构建REST API
    HTTP协议(一)
  • 原文地址:https://www.cnblogs.com/zhengbin/p/5633405.html
Copyright © 2011-2022 走看看