Java基础教程:反射基础

Java基础教程:反射基础

引入反射

反射是什么

  能够动态分析类能力的程序称为反射。反射是一种很强大且复杂的机制。

Class类

  在程序运行期间,Java运行时系统始终为所有对象维护一个被称为运行时的类型标识。这个对象追踪着每个对象所属的类。虚拟机利用运行时类的信息选择相应的方法执行

  我们可以通过专门的Java类访问这些信息。保存这些信息的类称为Class。

直接获取

Class cl1 =Date.class;
Class cl2 = Integer.class;

根据对象获得类信息

MyClass myClass;
.....
Class cl = myClass.getClass();
cl.getName(); //获得类名

调用静态方法forName获得类名相应的Class对象。

String className = 'java.util.Date';
Class cl = Class.forName(className);

注意:

  虚拟机为每一个类型管理一个Class对象。因此使用==可以实现两个类对象比较的操作

利用反射分析类的能力

API函数举例

  • getDeclaredFields():获得此类对象所表示的类或接口所声明的所有字段。
  • getFields():输出类以及其所继承的父类的 public 字段
  • getDeclaredMethods():获得此类对象所表示的类或接口所声明的所有方法,包括公共、保护、私有但不包括继承。
  • getDeclaredConstructors() :返回 Constructor 对象的一个数组,这些对象反映此 Class 对象表示的类声明的所有构造方法。
  • getName():获得此class所表示实体的名称。

示例:输出字段

  GetFields只会输出publicStr,GetDeclaredFields则会忽略访问控制符输出所有字段

public class MyClass{

    public String publicStr;
private String privateStr;
protected String protectedStr; public static void main(String[] args) {
//获取Class类
Class c1 = MyClass.class;
//调用GetFields()
for(Field field : c1.getFields()){
System.out.println(field.getName());
}
System.out.println("=============");
//调用GetDeclaredFields()
for(Field field : c1.getDeclaredFields()){
System.out.println(field.getName());
}
}
}

示例:输出方法

  getDeclaredMethods只会输出本类的所有方法!

public class MyClass{

    public void publicFun(int a){}
private void privateFun(int a){}
protected void protectFun(int a){} public static void main(String[] args) {
//获取Class类
Class c1 = MyClass.class; //2.1 获取所有 public 访问权限的方法
//包括自己声明和从父类继承的
//Method[] mMethods = c1.getMethods(); //2.2 获取所有本类的的方法(不问访问权限)
Method[] mMethods = c1.getDeclaredMethods(); //3.遍历所有方法
for (Method method :
mMethods) {
//获取并输出方法的访问权限(Modifiers:修饰符)
int modifiers = method.getModifiers();
System.out.print(Modifier.toString(modifiers) + " ");
//获取并输出方法的返回值类型
Class returnType = method.getReturnType();
System.out.print(returnType.getName() + " "
+ method.getName() + "( ");
//获取并输出方法的所有参数
Parameter[] parameters = method.getParameters();
for (Parameter parameter :
parameters) {
System.out.print(parameter.getType().getName()
+ " " + parameter.getName() + ",");
}
//获取并输出方法抛出的异常
Class[] exceptionTypes = method.getExceptionTypes();
if (exceptionTypes.length == 0) {
System.out.println(" )");
} else {
for (Class c : exceptionTypes) {
System.out.println(" ) throws "
+ c.getName());
}
}
}
}
}

  调用getDeclaredMethods输出结果:

  • public static void main( [Ljava.lang.String; arg0, )
  • protected void protectFun( int arg0, )
  • private void privateFun( int arg0, )
  • public void publicFun( int arg0, )

  调用getMethods输出结果:

  • public static void main( [Ljava.lang.String; arg0, )
  • public void publicFun( int arg0, )
  • public final void wait( ) throws java.lang.InterruptedException
  • public final void wait( long arg0,int arg1, ) throws java.lang.InterruptedException
  • public final native void wait( long arg0, ) throws java.lang.InterruptedException
  • public boolean equals( java.lang.Object arg0, )
  • public java.lang.String toString( )
  • public native int hashCode( )
  • public final native java.lang.Class getClass( )
  • public final native void notify( )
  • public final native void notifyAll( )

在运行时使用反射分析对象

API函数举例

  • setAccessible(boolean flag):为反射对象设置可访问标志。为true使得对象的私有属性也可以被查询和设置。
  • getField(): 返回一个 Field 对象,它反映此 Class 对象所表示的类或接口的指定公共成员字段。
  • get(Object obj):返回对象中用Field对象表示的域值。
  • set(Object obj,Object newValue):用一个新值设置Obj对象中的Field对象表示的域。

示例:获取并设置对象的私有字段值

public class MyClass{

    private String context = "Hello World";

    public static void main(String[] args) {
//获取Class类
Class c1 = MyClass.class;
try {
//构建对象
Object myClass = c1.newInstance();
//获取私有变量
Field context = c1.getDeclaredField("context");
//获取私有变量访问权
context.setAccessible(true);
//访问私有变量值
System.out.println(context.get(myClass));
//设置私有变量值
context.set(myClass, "Hi Boy!");
System.out.println(context.get(myClass));
} catch (Exception e) {
e.printStackTrace();
}
}
}

利用反射调用任意方法

API函数说明

   public Object invoke(Object obj,Object... args)

  • 参数:obj - 从中调用底层方法的对象args - 用于方法调用的参数
  • 返回:使用参数 args 在 obj 上指派该对象所表示方法的结果

示例:调用私有方法

public class NCConfgCliService {

    private void privateFun(int a){
System.out.println(a);
} public static void main(String[] args) {
//获取Class类
Class c1 = NCConfgCliService.class;
try {
//获取私有方法
Method privateMethod = c1.getDeclaredMethod("privateFun",int.class);
//创建类的实例
Object obj = c1.newInstance();
//调用私有方法
privateMethod.invoke(obj,10);
} catch (Exception e) {
e.printStackTrace();
}
}
}
上一篇:股票投资组合-前进优化方法(Walk forward optimization)


下一篇:黑马程序猿————Java基础日常笔记---反射与正則表達式