-->

signed

QiShunwang

“诚信为本、客户至上”

【注解与反射六】动态创建对象执行方法

2021/6/3 14:27:29   来源:

有了Class对象,能做什么?

  • 创建类的对象: 调用Class对象的newInstance()方法

    • 类必须有一个无参构造器
    • 类的构造器的访问权限要足够
    //获得Class对象
    Class c1 = Class.forName("注解与反射.User");
    //构造一个对象
    User user = (User) c1.newInstance();
    System.out.println(user);
    

难道没有无参构造器就不能创建对象了吗?只要在操作的时候明确调用类中的构造器,并将参数传递进去之后,才可以实例化操作。

  • 步骤如下:

    • 通过Class类的getDeclaredConstructor(Class … paramerTypes)取得本类的制定形参类型的都早起
    • 向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需要的各个参数
    • 通过Constructor实例化对象
    //获得Class对象
    Class c1 = Class.forName("注解与反射.User");
    //通过构造器创建一个对象
    Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
    User user2 = (User) constructor.newInstance("张三", 1, 18);
    System.out.println(user2);
    

调用指定的方法

通过反射,调用类中的方法,通过Method类完成。

  • 通过Class类的getMethod(String name,Class…parameterTypes)方法取得一个Method对象,并设置此方法操作时所需要的参数类型
  • 之后使用 Object invoke(Object obj,Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息。

在这里插入图片描述

  • Object invoke(Object obj,Object … args)
    • object对应原方法的返回值,若原方法无返回值,此时返回null
    • 若原方法若为静态方法,此时形参Object obj可为null
    • 若原方法形参列表为空,则Object[] args为null
    • 若原方法声明为private,则需要在调用此invoke()方法前,显式调用方法对象的setAccessible(true)方法,将可访问private的方法。
  • setAccessible
    • Method和Field、Constructor对象都有setAccessible()方法
    • setAccessible作用是启动和禁用访问安全检查的开关
    • 参数值为true则指示反射的对象在使用时应该取消Java语言访问检查
      • 提高反射效率。如果代码中必须使用反射,而该句代码需要频繁被调用,那么请设置为true
      • 使得原本无法访问的私有成员也可以访问
    • 参数值为false则指示反射的对象应该实施Java语言访问检查

代码演示:

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

//动态的创建对象
public class Test05 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //普通new一个对象
        //User user1 = new User();
        //System.out.println(user1);

        //获得Class对象
        //Class c1 = Class.forName("注解与反射.User");
        //构造一个对象
        //User user = (User) c1.newInstance();
        //System.out.println(user);

        //通过构造器创建一个对象
        //Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        //User user2 = (User) constructor.newInstance("张三", 1, 18);
        //System.out.println(user2);

		System.out.println("---------------------------------------------");
        Class c1 = Class.forName("注解与反射.User");
        User user3 = (User) c1.newInstance();

        //通过反射调用普通方法
        //通过反射获取一个方法
        Method setName = c1.getDeclaredMethod("setName", String.class);
        //激活:invoke(对象,"方法值")
        setName.invoke(user3,"李四");
        System.out.println(user3.getName());

        //通过反射操作属性
        User user4 = (User) c1.newInstance();
        Field name = c1.getDeclaredField("name");
        //User中的属性都是private的,默认不可访问
        //setAccessible():设置可访问性
        //setAccessible():将设个方法设置为true就可以直接访问了
        name.setAccessible(true);
        name.set(user4,"王五");
        System.out.println(user4.getName());
    }
}