网站首页 > 精选教程 正文
一、静态加载类和动态加载类
- 静态加载类:new创建对象,是静态加载类,在编译时刻就需要加载所有可能用到的类
- 动态加载类:在运行时加载 反射是一种动态加载类的机制
二、反射的优劣
- 优点:运行时类型的判断、动态加载类:提高了代码的灵活性,可以在不修改源码的情况下修改功能
- 缺点:存在性能瓶颈:需要进行安全性检查、反射相当于一系列解释操作,比直接的Java代码慢
三、通过反射了解泛型的本质
1、泛型只在编译期间生效
public class Test {
public static void main(String[] args) {
// 1、检验泛型擦除
List list1 = new ArrayList();
List<String> list2 = new ArrayList<String>();
System.out.println(list1.getClass()==list2.getClass());
}
}
运行结果:true
2、集合泛型是为了类型检查,避免错误输入
List<String> list2 = new ArrayList<String>();
list2.add("a");
list2.add(20);
编译错误:int无法转换为java.lang.String
3、可以通过反射绕过泛型的检查,添加不同类型的元素
List<String> list2 = new ArrayList<String>();
list2.add("a");
Class<?> c = list2.getClass();
Method method = c.getDeclaredMethod("add",Object.class);
method.invoke(list2,20);
System.out.println(list2.size());
运行结果:2
可以看到,由于泛型的类型检查只在编译时有效,利用反射的动态加载原理,可以绕过泛型的检查,往集合里添加不同类型的元素
四、反射的应用
1、加载数据库驱动
// DriverManager.registerDriver(new com.mysql.cj.jdbc.Driver());
Class.forName("com.mysql.cj.jdbc.Driver");
2、xml或properties等配置文件加载
Spring通过XML配置模式装载Bean的过程
- 将程序中所有XML或properties配置文件加载入内存
- Java类里面解析xml或者properties里面的内容,得到对应实体类的字节码字符串以及相关的属性信息
- 使用反射机制,根据这个字符串获得某个类的Class实例
- 动态配置实例的属性
配置文件
className=com.example.reflectdemo.TestInvoke
methodName=printlnState
实体类
public class TestInvoke {
private void printlnState(){
System.out.println("I am fine");
}
}
解析配置文件内容
// 解析xml或properties里面的内容,得到对应实体类的字节码字符串以及属性信息
public static String getName(String key) throws IOException {
Properties properties = new Properties();
FileInputStream in = new FileInputStream("D:\IdeaProjects\AllDemos\language-specification\src\main\resources\application.properties");
properties.load(in);
in.close();
return properties.getProperty(key);
}
利用反射获取实体类的Class实例,创建实体类的实例对象,调用方法
public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, IOException, ClassNotFoundException, InstantiationException {
// 使用反射机制,根据这个字符串获得Class对象
Class<?> c = Class.forName(getName("className"));
System.out.println(c.getSimpleName());
// 获取方法
Method method = c.getDeclaredMethod(getName("methodName"));
// 绕过安全检查
method.setAccessible(true);
// 创建实例对象
TestInvoke testInvoke = (TestInvoke)c.newInstance();
// 调用方法
method.invoke(testInvoke);
}
运行结果:
作者:bkpp976
链接:https://juejin.cn/post/7042974349883604999
来源:稀土掘金
猜你喜欢
- 2024-11-13 十年Java工程师写下的Java反射使用总结
- 2024-11-13 一起来了解Java中的反射 java反射的步骤原理
- 2024-11-13 Java基础:反射机制,你用得多吗 java反射机制有什么用
- 2024-11-13 每日一点涨薪小知识-JAVA反射 java反射的理解
- 2024-11-13 不是吧,还有人连Java最强大的技术之一:反射还没搞懂?赶紧码住
- 2024-11-13 浅谈java中的反射(恭喜FPX) java里的反射是什么
- 2024-11-13 实操讲解Java的反射机制,你要是再看不懂,神仙都没救了
- 2024-11-13 两分钟学会Java反射 java 反射
- 2024-11-13 三石说:java基础 之 反射 java反射的基石
- 2024-11-13 0202年了,还有人不懂Java反射机制?
你 发表评论:
欢迎- 最近发表
- 标签列表
-
- nginx反向代理 (57)
- nginx日志 (56)
- nginx限制ip访问 (62)
- mac安装nginx (55)
- java和mysql (59)
- java中final (62)
- win10安装java (72)
- java启动参数 (64)
- java链表反转 (64)
- 字符串反转java (72)
- java逻辑运算符 (59)
- java 请求url (65)
- java信号量 (57)
- java定义枚举 (59)
- java字符串压缩 (56)
- java中的反射 (59)
- java 三维数组 (55)
- java插入排序 (68)
- java线程的状态 (62)
- java异步调用 (55)
- java中的异常处理 (62)
- java锁机制 (54)
- java静态内部类 (55)
- java怎么添加图片 (60)
- java 权限框架 (55)
本文暂时没有评论,来添加一个吧(●'◡'●)