【Java】反 射 思 想

目录

        一、反射思想:Java反射机制

        二、获取一个类的信息

        2.1、获取Class 类的对象

        2.2、反射获取构造方法

        2.3、反射获取成员变量

        2.4、反射获取成员方法

        三、反射思想的使用

        3.1、获取构造方法的使用示例①

        3.2、获取构造方法的使用示例②

        3.3、获取成员变量的使用示例

扫描二维码关注公众号,回复: 13563481 查看本文章

        3.4、获取成员方法的使用示例

        3.5、【反射练习】越过泛型检查

        3.6、【反射练习】运行配置文件指定内容


        一、反射思想:Java反射机制

        是指在运行时去获取一个类的变量和方法信息。然后通过获取到的信息来创建对象,调用方法的一种机制。由于这种动态性,可以极大的增强程序的灵活性,程序不用在编译期就完成确定,在运行期仍可以扩展。

        二、获取一个类的信息

        2.1、获取Class 类的对象

我们要想通过反射去使用一个类,首先我们要获取到该类的字节码文件对象,也就是类型为Class类型的对象。 这里我们提供三种方式获取Class类型的对象。
使用类的class 属性来获取该类对应的Class对象。 举例:Student.class将会返回Student 类对应的Class 对象。
调用对象的getClass() 方法,返回该对象所属类对应的Class 对象。
使用Class 类中的静态方法forName(String className),该方法需要传入字符串参数,该字符串参数的值是某个类的全路径,也就是完整包名的路径。

        2.2、反射获取构造方法

Constructor<?>[] getConstructors​()  返回所有公共构造方法对象的数组。
Constructor<?>[] getDeclaredConstructors​()  返回所有构造方法对象的数组。
Constructor<T> getConstructor​(Class<?>... parameterTypes)      返回单个公共构造方法对象。
Constructor<T> getDeclaredConstructor​(Class<?>... parameterTypes)  返回单个构造方法对象。
Constructor类中用于创建对象的方法:
T newInstance​(Object... initargs)  根据指定的构造方法创建对象。

        2.3、反射获取成员变量

Field[] getFields​()  返回所有公共成员变量对象的数组。
Field[] getDeclaredFields​()  返回所有成员变量对象的数组。
Field getField​(String name)  返回单个公共成员变量对象。
Field getDeclaredField​(String name)  返回单个成员变量对象。
Field类中用于给成员变量赋值的方法:
void set​(Object obj, Object value)  给obj对象的成员变量赋值为value。

        2.4、反射获取成员方法

Method[] getMethods​()  返回所有公共成员方法对象的数组,包括继承的。
Method[] getDeclaredMethods​()  赶回所有成员方法对象的数组,不包括继承的。
Method getMethod​(String name, Class<?>... parameterTypes)  返回单个公共成员方法对象。
Method getDeclaredMethod​(String name, Class<?>... parameterTypes)  返回单个成员方法对象。
Method类中用于调用成员方法的方法:
Object invoke​(Object obj, Object... args)  在具有指定参数的指定对象上调用此方法对象表示的基础方法。  

        三、反射思想的使用

        已有自定义类Student,代码如下:

package com.myReflect.demo1;
public class Student {
    //成员变量:一个私有,一个默认,一个公共
    private String name;
    int age;
    public String address;

    //构造方法:一个私有,一个默认,两个公共
    public Student() {
    }

    private Student(String name) {
        this.name = name;
    }

    Student(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Student(String name, int age, String address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

    //成员方法:一个私有,四个公共
    private void function() {
        System.out.println("function");
    }

    public void method1() {
        System.out.println("method");
    }

    public void method2(String s) {
        System.out.println("method:" + s);
    }

    public String method3(String s, int i) {
        return s + "," + i;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", address='" + address + '\'' +
                '}';
    }
}

        3.1、获取构造方法的使用示例①

  •  通过反射实现如下操作:
        Student s = new Student("林青霞", 30, 西安");
        System.out.println(s);
//获取Class对象
Class<?> c = Class.forName("com.myReflect.demo1.Student");

Constructor<?> con = c.getConstructor(String.class, int.class, String.class);
//基本数据类型也可以通过.class得到对应的Class类型

//新建对象
Object obj = con.newInstance("林青霞", 30, "西安");
System.out.println(obj);

        3.2、获取构造方法的使用示例②

  • 通过反射实现如下操作:
        Student s = new Student("林青霞");        
        System.out.println(s); 
//获取Class对象
Class<?> c = Class.forName("com.myReflect.demo1.Student");

Constructor<?> con = c.getDeclaredConstructor(String.class);

//暴力反射,public void setAccessible​(boolean flag):值为true时,取消访问检查
con.setAccessible(true);

Object obj = con.newInstance("林青霞");//此时对应的有参构造方法是private的,所以需要暴力反射
System.out.println(obj);

        3.3、获取成员变量的使用示例

  • 练习:通过反射实现如下操作
        Student s = new Student();
        s.name = "林青霞";
        s.age = 30;
        s.address = "西安";
        System.out.println(s); 
// 1、获取构造对象
Class<?> c = Class.forName("com.myReflect.demo1.Student");

// 2、新建该对象的无参方法
Constructor<?> con = c.getDeclaredConstructor();
Object obj = con.newInstance();
System.out.println(obj);

// 3、赋值:s.name = "林青霞";
//Field nameFiled = c.getField("name");//报错:NoSuchFieldException
Field nameFiled = c.getDeclaredField("name");//getDeclaredField():返回单个成员变量对象。
nameFiled.setAccessible(true);//暴力反射:setAccessible(true)取消访问检查
nameFiled.set(obj,"林青霞");//给obj的nameFiled成员变量赋值
System.out.println(obj);

// 4、赋值:s.age = 30;
Field ageFiled = c.getDeclaredField("age");
ageFiled.setAccessible(true);
ageFiled.set(obj,30);
System.out.println(obj);

// 5、赋值:s.address = "西安";
Field addressFiled = c.getDeclaredField("address");
addressFiled.setAccessible(true);
addressFiled.set(obj,"西安");
System.out.println(obj);

        3.4、获取成员方法的使用示例

  •  练习:通过反射实现如下操作
        Student s = new Student();
        s.method1();
        s.method2("林青霞");
        String ss = s.method3("林青霞",30);
        System.out.println(ss);
        s.function();
// 1、获取Class对象
Class<?> c = Class.forName("com.myReflect.demo1.Student");

// 2、新建该对象的无参构造方法
Constructor<?> con = c.getConstructor();
Object obj = con.newInstance();

// 3.1、s.method1();
Method m1 = c.getMethod("method1");
m1.invoke(obj);

// 3.2、s.method2("林青霞");
Method m2 = c.getMethod("method2", String.class);
m2.invoke(obj,"林青霞");

//3.3、String ss = s.method3("林青霞",30); System.out.println(ss);
Method m3 = c.getMethod("method3", String.class, int.class);
m3.invoke(obj,"林青霞",30);

// 3.4、s.function();
//Method mf = c.getMethod("function");//报错:NoSuchMethodException
Method mf = c.getDeclaredMethod("function");
mf.setAccessible(true);
mf.invoke(obj);

        3.5、【反射练习】越过泛型检查

  •  现有一个ArrayList<Integer>集合,要求在这个集合中添加一个字符串数据,如何实现? 
ArrayList<Integer> arrayList = new ArrayList<>();

Class<? extends ArrayList> cArray = arrayList.getClass();
Method mAdd = cArray.getMethod("add", Object.class);
mAdd.invoke(arrayList,"hello");
mAdd.invoke(arrayList,"world");
mAdd.invoke(arrayList,"Java");

System.out.println(arrayList);

        3.6、【反射练习】运行配置文件指定内容

  • 通过配置文件(class.txt)运行类(如Student、Teacher)中的方法

         主函数中的代码:

//写入基础数据
Properties prop = new Properties();
prop.setProperty("className","com.myReflect.demo4.Teacher");
prop.setProperty("methodName","teach");

FileWriter fw = new FileWriter("myReflect\\src\\com\\myReflect\\demo4\\class.txt");
prop.store(fw,null);
fw.close();

//加载数据
FileReader fr = new FileReader("myReflect\\src\\com\\myReflect\\demo4\\class.txt");
prop.load(fr);
fr.close();
String className = prop.getProperty("className");
String methodName = prop.getProperty("methodName");

Class<?> c = Class.forName(className);
Object obj = c.newInstance();
Method method = c.getMethod(methodName);
method.invoke(obj);

        自定义Teacher类:

package com.myReflect.demo4;

public class Teacher {
    public void teach() {
        System.out.println("用爱成就学员");
    }
}

         自定义Student类:

package com.myReflect.demo4;

public class Student {
    public void study() {
        System.out.println("好好学习天天向上");
    }
}

        class.txt中的内容:

methodName=teach
className=com.myReflect.demo4.Teacher

 

猜你喜欢

转载自blog.csdn.net/weixin_60808029/article/details/121613070
今日推荐