java 基础 反射

174 阅读4分钟

反射

Java的反射(reflection)机制是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法。在运行时调用任意一个对象的方法;生成动态代理。

在运行时判断任意一个对象所属的类。
在运行时构造任意一个类的对象。
在运行时判断任意一个类所具有的成员变量和方法。
在运行时调用任意一个对象的方法

比如我们使用的jdbc就是一个反射的列子:

Class.forName("com.mysql.jdbc.Driver");

Java反射机制的实现要借助于4个类:Class,Constructor,Field,Method;

其中class代表的是类对象,Constructor-类的构造器对象,Field-类的属性对象,Method-类的方法对象,通过这四个对象我们可以粗略的看到一个类的各个组成部分。其中最核心的就是Class类,它是实现反射的基础,它包含的方法我们在第一部分已经进行了基本的阐述。应用反射时我们最关心的一般是一个类的构造器、属性和方法,下面我们主要介绍Class类中针对这三个元素的方法:

1、得到构造器的方法

Constructor getConstructor(Class[] params) -- 获得使用特殊的参数类型的公共构造函数, 
 
Constructor[] getConstructors() -- 获得类的所有公共构造函数 
 
Constructor getDeclaredConstructor(Class[] params) -- 获得使用特定参数类型的构造函数(与接入级别无关) 
 
Constructor[] getDeclaredConstructors() -- 获得类的所有构造函数(与接入级别无关) 

2、获得字段信息的方法

Field getField(String name) -- 获得命名的公共字段 
 
Field[] getFields() -- 获得类的所有公共字段 
 
Field getDeclaredField(String name) -- 获得类声明的命名的字段 
 
Field[] getDeclaredFields() -- 获得类声明的所有字段 

3、获得方法信息的方法

Method getMethod(String name, Class[] params) -- 使用特定的参数类型,获得命名的公共方法 
 
Method[] getMethods() -- 获得类的所有公共方法 
 
Method getDeclaredMethod(String name, Class[] params) -- 使用特写的参数类型,获得类声明的命名的方法 
 
Method[] getDeclaredMethods() -- 获得类声明的所有方法 

在程序开发中使用反射并结合属性文件,可以达到程序代码与配置文件相分离的目的

如果我们想要得到对象的信息,一般需要“引入需要的‘包.类’的名称——通过new实例化——取得实例化对象”这样的过程。使用反射就可以变成“实例化对象——getClass()方法——得到完整的‘包.类’名称”这样的过程。

正常方法是通过一个类创建对象,反射方法就是通过一个对象找到这个类

package com.blueearth.bewemp.doc.config;

/**
 * @user:
 * @date:2021/1/18
 * @Description:
 */
public class Fruits {
    private String color;
    private double price;
    private String size;
    public  String  obs;

    public String getObs() {

        return obs;
    }

    public void setObs(String obs) {
        this.obs = obs;
    }

    public String getColor() {
        return color;
    }

    public void setColor(String color) {
        this.color = color;
    }

    public double getPrice() {
        return price;
    }

    public void setPrice(double price) {
        this.price = price;
    }

    public String getSize() {
        return size;
    }

    public void setSize(String size) {
        this.size = size;
    }
}

/// 测试代码
package com.blueearth.bewemp.doc.config;

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

/**
 * @user: kaysanshi
 * @date:2021/1/18
 * @Description: 测试反射的列子
 */
public class TestReflection {
    public static void main(String [] args) throws Exception {
        // 正常的使用
        Fruits fruits = new Fruits();
        fruits.setPrice(10);
        System.out.println(fruits.getPrice());
        // 使用反射
        Class<?> aClass = Class.forName("com.blueearth.bewemp.doc.config.Fruits");
        // 获取命名的公共方法
        Method setPrice = aClass.getMethod("setPrice", double.class);

        // 获取所有的公共属性getFields()(无法获取私有属性)
        System.out.println("获取共有属性");
        Field[] fields = aClass.getFields();
        for(Field field : fields) {
            System.out.println(field.getName());
        }
        System.out.println("获取私有属性");
        // getDeclaredFields()(获取私有属性)
        Field[] dfields = aClass.getDeclaredFields();
        for(Field field : dfields){
            System.out.println(field.getName());
        }
        // 获取构造器
        Constructor fruitsConstructor = aClass.getConstructor();
        // 通过构造器的newInstance()获取反射类对象
        Object obj = fruitsConstructor.newInstance();
        // 利用 invoke方法进行调用方法
        setPrice.invoke(obj,13);
    }
}

方法名为diff,输入参数为两个同类的对象。输出为一个List,其中存放了两个对象的不同的属性的属性名称。

假设User对象,包含name\age\phone三个参数。

// 不使用反射
    List<String> diff(User user1,User user2){
        List<String> result  = new ArrayList<>();
        if(user1.getName() == null && user2.getName() !=null || !user1.getName().equals(user2.getName())){
            result.add("name");
        }
        if(user1.getAge() == null && user2.getAge() !=null || !user1.getAge().equals(user2.getAge())){
            result.add("age");
        }
        if(user1.getPhone() == null && user2.getPhone() !=null || !user1.getPhone().equals(user2.getPhone())){
            result.add("phone");
        }
    }
    // 使用反射
    List<String> diff(Object obj1,Object obj2){
        try{
            Class<?> clazz1 = obj1.getClass();
            Class<?> clazz2 = obj2.getClass();
            if(clazz1.equals(clazz2)){
                List<String> result = new ArrayList<>();
                for(Field feild: clazz1.getDeclaredFields()){
                    feild.setAccessible(true);
                    Object obj1= feild.get(obj1);
                    Object obj2=feild.get(obj2);
                    if(value1==null && value2!=null || !value1.equals(value2)){
                        result.add(feild.getName());
                    }
                }
                return result;
            }
            return null;
        }catch (Exception e){
            e.printStackTrace();
            return null;
        }
    }

该方法的优点是可以处理任何类的两个对象,而不需要关心它的属性,十分通用

反射如何创建对象

1.通过获取构造器 getConstructor() 然后 通过构造器的newInstance()获取反射类对象。

2.通过类对象调用newInstance()方法。例如

// 使用反射
Class<?> aClass = Class.forName("com.blueearth.bewemp.doc.config.Fruits");
Object o = aClass.newInstance();
System.out.println(o);

Q&A 反射的面试题

blog.csdn.net/qq_37875585…