java反射机制 入门 理解

一、java中生成对象的方式,本人所能想到的有以下几种(以 com.knight.Employer为例)

1、使用new方式,这每个javaer都会的

   Employer e=new Employer();

2、使用克隆方式(clone),此方式要求Employer实现cloneable接口

   Employer e2=(Employer)e1.clone();

3、序列化方式,此方式要求Employer实现Serializable接口

   ObjectInputStream objIn=new ObjectInputStream(in);//in为InputStream实例

   Employer e=(Employer)objIn.readObject();

4、反射

  反射可以有很多变体,通常有以下几种

  Class c=Class.forName("com.knight.Employer");

  Employer e= (Employer)c.newInstance();

  或者

  ClassLoader loader=Huma.class.getClassLoader();

  Class c=loader.loadClass("com.knight.Employer");

  Employer e= (Employer)c.newInstance();

  此两种方式都要求Employer有公有无参构造函数。

  还有另外一种通过Constructor进行构造,以下详述

5、直接操作字节码,这种方式需要对class文件及虚拟机机制有深刻理解,不是一般的难

 

二、java反射api简介

以下与Huma类为例

package com.knight.test;
import java.io.ObjectInputStream;
import java.io.Serializable;
import java.lang.reflect.Constructor;
public class Huma {
 public String name;
 private int age;
 private void log(){
  System.out.println("log invoke");
 }
 public Huma() {
 
 }
 
 public Huma(String name,int age) {
   this.name=name;
   this.age=age;
 }

 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;
 }
}

 

java中反射机制主要使用以下几个类

Class,Constructor,Field,Method,AccessibleObject

1、Class的用法

Class可以获取关于类中所有方法,属性的信息

    Field[] getFields()//获取Huma中声明的公有属性,包括从父类继承的

    Field[] getDeclaredFields()//获取Huma中声明所有属性(包括私有属性),但不包含任何父类继承的属性

注意有无declared的区别,以下不在罗嗦

 

     Field getField(String name)//获取名称为name的属性

以此类似,Class可以获取其对于的Constructor,Method信息

再举一个方法

     Method getDeclaredMethod(String name, Class<?>... parameterTypes);此方法获取成员函数名字为name,函数的参数类型为Class<?>... parameterTypes的函数,其中parameterTypes为null或者为空数组时,表示该函数没有形参

 

 

 

public Object newInstance() //生成该类的一个对象,需有无参构造函数

public static Class forName(String className)//生成className指定的类的Class实例

2、AccessibleObject

   AccessibleObject是一个接口,Constructor,Field,Method均实现了该接口

  

public void setAccessible(boolean flag);该方法设置可访问性,在类的成员为私有的情况下

设置flat为true将可以通过反射修改成员的值,否则会发生异常

3、Field类

 public Object get(Object o);获取该Field实例对应对象o的Field的值

 public void set(Object obj, Object value)  ;设置该Field对应的对象obj的Field属性的值为value

 

3、Method类

    Class<?>[] getParameterTypes() ;返回该Method的形参类型,按声明顺序

    Class<?> getReturnType()   ;返回该Method的返回类型

    Class<?>[] getExceptionTypes() ;返回该Method声明抛出的异常

    Object invoke(Object obj, Object... args) ;执行该Method,obj是该Method所属对象的一个实例

                                                 //args是一个对象数组,表示该Method参数数组实参 
   其中args为null或者为空数组时,表示该函数没形参,当obj为null时,表示该函数为静态方法

4、Constructor类

    Class<T> getDeclaringClass();返回该Constructor所表示的Class对象

    Class<?>[] getParameterTypes() ;返回该Constructor表示的形参类型数组

    T newInstance(Object... initargs) ;生成该Constructor表示的对象的一个实例,initargs为形参数组
  

三、例子

  //  生成Huma实例(带参数)

    Constructor c=Huma.class.getDeclaredConstructor(new Class[]{String.class,int.class}); 

    Huma huma=c.newInstance(new Object[]{"zhang san",18});

 

 //获取Field,并改变其值

   Field nameField=Huma.class.getField("zhangsan");

   nameField.setsetAccessible(true);//注意name属性私有,所以在这里调用,参数为true  

   nameField.set("lisi");//此时name已经变为lisi了,上一句不调用的的话,此句将抛出异常

 

//获取方法并调用

   Method methodSetName=Huma.class.getMethod("setName",new Class[]{String.class});

   methodSetName.invoke(huma,new Object[]{"wangwu"});//相当于 huma.setName("wangwu");

 

 最后需要注意的地方就是反射生成的类,修饰符应该是public的,否则,在其他地方调用时可能会出现问题

,例如 在一个 com.abc.Test的main方法中调用 Class.forName("com.knight.Huma"),并且Huma的修饰符不是public时,将抛出异常

你可能感兴趣的:(java,C++,c,虚拟机,C#)