什么是反射?    

反射是动态获取信息(属性和行为),并动态调用方法的一种机制

补充:  反射介绍     

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法        对于任意一个对象,都能够调用它的任意一个方法        这种动态获取的以及动态调用对象的方法的功能称为java语言的反射机制.        简单来说, 就可以把.class文件比做动物的尸体, 而反射技术就是对尸体的一种解剖.   通过反射技术, 我们可以拿到该字节码文件中所有的东西, 例如成员变量, 成员方法, 构造方法, 而且还包括私有

什么是Java反射?        

java反射是程序运行过程中,动态获取任意一个类的属性和行为,然后可以选择创建对象,并执行方法。也可以改变其属性。所以Java被视为动态语言的关键性质之一。

反射的好处?    

1、增强程序的灵活性,

2、程序不用在编译期就完成确定,

3、在运行期仍然可以扩展

反射操作步骤    

1、首先拿到加载类的字节码文件(1类名.class    2创建了对象,拿对象.getclass()     3Class.forname("全类名(src下的路径)")) 

2、然后通过字节码文件获取其构造方法  :字节码文件.getXXX获取构造方法的字节码文件(getConstructors())    (如果需要越过权限(public不需要,但是protected和defult和private可能需要)检查需要使用).setAccessible(true);

3、然后拿到构造方法的对象使用newinstance()方法,创建了对象       

4、如果要拿成员属性的话 那就getField        如果要对成员属性赋值的话,使用成员属性对象.set(需要赋的值)        

5、如果要拿方法的话,那就getXXXMethord(),        

6、要执行该方法的话,就使用拿到方法对象.invoke(这里放个对象)。

获取构造方法对象的方法:

通过构造方法对象创建类的对象的方法:

通过类对象获取成员变量的方法:

通过类的对象给Field辅助的方法

通过类对象获取成员方法对象的方法

  类的对象执行方法的方法

反射注意事项       

直接拿类对象可以直接使用newinstance方法创建对象,但是创建对象的过程是调用其无参数构造方法,(加入有了带参构造,一旦JVM就不会给类添加无参数构造方法,就会出现问题)   通过类对象去拿构造方法,可以使用构造方法对象使用newinstance方法创建对象,但是为了避免出现权限检测,一般使用暴力反射(setAcesssible(true))来解决。

newInstance和new都可以创建对象,他们两有什么区别呢?(Constructor.newInstance,一个是Class.newInstance

1、类的加载方式不同      

在执行Class.forName("a.class.Name")时,JVM会在classapth中去找对应的类并加载,这时JVM会执行该类的静态代码段。在使用newInstance()方法的时候,必须保证这个类已经加载并且已经连接了,而这可以通过Class的静态方法forName()来完成的。     使用关键字new创建一个类的时候,这个类可以没有被加载,一般也不需要该类在classpath中设定,但可能需要通过classlaoder来加载。   

2、所调用的构造方法不尽相同     

new关键字能调用任何构造方法。     newInstance()只能调用无参构造方法。   

3、执行效率不同     

new关键字是强类型的,效率相对较高。

newInstance()是弱类型的,效率相对较低。

既然使用newInstance()构造对象的地方通过new关键字也可以创建对象,为什么又会使用newInstance()来创建对象呢?

假设定义了一个接口Door,开始的时候是用木门的,定义为一个类WoodenDoor,在程序里就要这样写 Door door = new WoodenDoor() 。假设后来生活条件提高,换为自动门了,定义一个类AutoDoor,这时程序就要改写为 Door door = new AutoDoor() 。虽然只是改个标识符,如果这样的语句特别多,改动还是挺大的。于是出现了工厂模式,所有Door的实例都由DoorFactory提供,这时换一种门的时候,只需要把工厂的生产模式改一下,还是要改一点代码。

而如果使用newInstance(),则可以在不改变代码的情况下,换为另外一种Door。具体方法是把Door的具体实现类的类名放到配置文件中,通过newInstance()生成实例。这样,改变另外一种Door的时候,只改配置文件就可以了。

示例代码如下:String className = 从配置文件读取Door的具体实现类的类名;Door door = (Door) Class.forName(className).newInstance();   

再配合依赖注入的方法,就提高了软件的可伸缩性、可扩展性。

一些关于反射的网站 http://bbs.itheima.com/thread-384999-1-1.html

Hutool 反射工具 ReflectUtil

import cn.hutool.core.util.ReflectUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Date;
import java.util.Objects;

@Slf4j
public class UserVO {

    public static String abc = "123";

    @Value("woshizhujie")
    public static Date date = new Date();

    public static void main(String[] args) throws Exception {

        // 获取类的所有方法
        Method[] methods = ReflectUtil.getMethods(UserVO.class);
        log.info("获得方法数量:{}个", methods.length);

        for (Method method : methods) {
            log.info("方法名是:{} , 返回值类型是:{} , 方法入参:{}", method.getName(), method.getReturnType().getName(), method.getParameterTypes());
        }

        System.out.println("\n");

        // 获取指定方法名的方法
        Method main = ReflectUtil.getMethod(UserVO.class, "finalize");
        log.info(Objects.nonNull(main) ? main.getName() + "已存在" : "方法名不存在");

        // 获取类的字段列表
        Field[] fields = ReflectUtil.getFields(UserVO.class);
        for (Field field : fields) {
            log.info(field.getName() + " : " + field.getType());
        }

        System.out.println("\n");

        // 构造对象
        UserVO userVO = ReflectUtil.newInstance(UserVO.class);
        log.info("构造对象:{}", userVO);

    }

结果打印一下吧

2024-01-23 14:49:35.819|INFO ||||||[main]|c.x.a.UserVO - [main,24] - 获得方法数量:13个
2024-01-23 14:49:35.821|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:main , 返回值类型是:void , 方法入参:[class [Ljava.lang.String;]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:finalize , 返回值类型是:void , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:wait , 返回值类型是:void , 方法入参:[long, int]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:wait , 返回值类型是:void , 方法入参:[long]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:wait , 返回值类型是:void , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:equals , 返回值类型是:boolean , 方法入参:[class java.lang.Object]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:toString , 返回值类型是:java.lang.String , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:hashCode , 返回值类型是:int , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:getClass , 返回值类型是:java.lang.Class , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:clone , 返回值类型是:java.lang.Object , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:registerNatives , 返回值类型是:void , 方法入参:[]
2024-01-23 14:49:35.822|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:notify , 返回值类型是:void , 方法入参:[]
2024-01-23 14:49:35.823|INFO ||||||[main]|c.x.a.UserVO - [main,27] - 方法名是:notifyAll , 返回值类型是:void , 方法入参:[]


2024-01-23 14:49:35.827|INFO ||||||[main]|c.x.a.UserVO - [main,34] - finalize已存在
2024-01-23 14:49:35.828|INFO ||||||[main]|c.x.a.UserVO - [main,39] - log : interface org.slf4j.Logger
2024-01-23 14:49:35.828|INFO ||||||[main]|c.x.a.UserVO - [main,39] - abc : class java.lang.String
2024-01-23 14:49:35.828|INFO ||||||[main]|c.x.a.UserVO - [main,39] - date : class java.util.Date


2024-01-23 14:49:35.829|INFO ||||||[main]|c.x.a.UserVO - [main,46] - 构造对象:com.zanglikun.archimaster.UserVO@709ba3fb
特殊说明:
上述文章均是作者实际操作后产出。烦请各位,请勿直接盗用!转载记得标注原文链接:www.zanglikun.com
第三方平台不会及时更新本文最新内容。如果发现本文资料不全,可访问本人的Java博客搜索:标题关键字。以获取全部资料 ❤