java反射机制

/**
 * 获取class类对象的三种方式
 * 1.通过类名获取      类名.class
 * 2.通过对象获取      对象名.getClass()
 * 3.通过全类名获取    Class.forName(全类名)
 */
@Test
public void test1() {
    //第一种通过类名获取
    Class clas = Person.class;
    System.out.println(clas);
}
@Test
public void test2(){
    //获取对象名.getClass()
   Object object=  new Person();
   Class class1= object.getClass();
    System.out.println(class1);
}
@Test
public void test3() throws ClassNotFoundException, IllegalAccessException, InstantiationException {
    //通过全类名获取 Class.forName(全类名)
    String classname="com.bootdo.common.config.collection.vo.Person";
    Class class3= Class.forName(classname);
    System.out.println(class3);
    //利用Class对象的newInstance方法创建一个类的实例
    Object object=class3.newInstance();

    System.out.println(object);

}

可以看出确实是创建了一个Person实例
  但是Person类有两个构造方法,到底是调用的哪一个构造方法呢

  实际调用的是类的无参数的构造器。所以在我们在定义一个类的时候,定义一个有参数的构造器,作用是对属性进行初始化,还要写一个无参数的构造器,作用就是反射时候用。

  一般地、一个类若声明一个带参的构造器,同时要声明一个无参数的构造器

2.反射

Reflection(反射)是Java被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的內部信息,并能直接操作任意对象的内部属性及方法。

  Java反射机制主要提供了以下功能:

  • 在运行时构造任意一个类的对象
  • 在运行时获取任意一个类所具有的成员变量和方法
  • 在运行时调用任意一个对象的方法(属性)
  • 生成动态代理
/**
  *
  * @param obj: 方法执行的那个对象.
  * @param methodName: 类的一个方法的方法名. 该方法也可能是私有方法.
  * @param args: 调用该方法需要传入的参数
  * @return: 调用方法后的返回值
  *
  */
 public Object invoke(Object obj,String methodName,Object ... args  ) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        Class[] parameterTypes =new Class[args.length];
     for (int i = 0; i < parameterTypes.length; i++) {
         parameterTypes[i] = args[i].getClass();
         System.out.println(parameterTypes[i]);
     }

     Method method= obj.getClass().getDeclaredMethod(methodName,parameterTypes);
     method.setAccessible(true);
     return method.invoke(obj,args);
 }

 /**
  * @param className: 某个类的全类名
  * @param methodName: 类的一个方法的方法名. 该方法也可能是私有方法.
  * @param args: 调用该方法需要传入的参数
  * @return: 调用方法后的返回值
  */
public Object invoke1(String className,String methodName,Object...args){
    try {
         Object object= Class.forName(className).newInstance();
         return this.invoke(object,methodName,args);
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    }
    return null;
}
//调用的代码
@Test
public void test6() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
     Person person=  new Person();
     //第一种的调用
     person.invoke(person,"test","2");
     //第二种的调用
     person.invoke1("com.bootdo.common.config.collection.vo.Person","test","2");
     //使用系统的方法调用
     //Object result = invoke("java.text.SimpleDateFormat", "format", new Date());
    //System.out.println(result);

}

反射小结

 1. Class: 是一个类; 一个描述类的类.

  封装了描述方法的 Method,

       描述字段的 Filed,

              描述构造器的 Constructor 等属性.
 
 2. 如何得到 Class 对象:
    2.1 Person.class
    2.2 person.getClass()
    2.3 Class.forName("com.atguigu.javase.Person")
  
 3. 关于 Method:
    3.1 如何获取 Method:
      1). getDeclaredMethods: 得到 Method 的数组.
      2). getDeclaredMethod(String methondName, Class ... parameterTypes)
  
    3.2 如何调用 Method
      1). 如果方法时 private 修饰的, 需要先调用 Method 的 setAccessible(true), 使其变为可访问
      2). method.invoke(obj, Object ... args);
  
  4. 关于 Field:
    4.1 如何获取 Field: getField(String fieldName)
    4.2 如何获取 Field 的值: 
      1). setAccessible(true)
      2). field.get(Object obj)
    4.3 如何设置 Field 的值:
      field.set(Obejct obj, Object val)
  5. 反射和泛型.
    6.1 getGenericSuperClass: 获取带泛型参数的父类, 返回值为: BaseDao<Employee, String>
    6.2 Type 的子接口: ParameterizedType
    6.3 可以调用 ParameterizedType 的 Type[] getActualTypeArguments() 获取泛型参数的数组.

 

 

 

转载于:https://my.oschina.net/u/3653755/blog/2051958

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值