什么是反射?
反射是动态获取信息(属性和行为),并动态调用方法的一种机制
补充: 反射介绍
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
第三方平台不会及时更新本文最新内容。如果发现本文资料不全,可访问本人的Java博客搜索:标题关键字。以获取最新全部资料 ❤
评论(0)