java f反射_Java基础篇-反射

96b991716447a02597d333db28e37790.png

编程中直接用到反射的地方很少,不过研究Spring、mybatis等框架会发现,反射是实现这些框架的基础。让我们一块看下反射和它的应用吧。

1、什么是反射2、class加载方法3、反射获取属性和方法

本文中包含代码较多,最好可以动手写一下,更容易理解

一、反射的定义

反射机制是指在程序的运行状态中,可以构造任意类的对象,了解任意对象所属的类,了解任意类的成员变量和方法,可以调用任意对象的属性和方法。

即,对类和实例化对象可以动态进行以下操作:

1、通过类构造实例化对象2、通过类读取类包含的属性方法3、通过实例化对象获取类的信息4、通过实例化对象操作对象的所有属性和方法

这么方便的操作,为什么在日常编码中很少直接使用呢,反射有如下问题:

1、性能差,反射是一种解释性操作,直接通知JVM进行的操作,无法进行代码优化,比直接使用类和实例化对象要慢的多。

2、安全问题,通过反射可以执行实例化对象所有的属性方法,包括private的方法,会引起一些安全错误

3、可读维护性差,使用反射相对直接调用,不能直观的看到调用的属性方法,出问题了,代码的可读、调试和维护性会很差

4、抽象性,对象改变了,反射中的逻辑可能无法感知,引起一些莫名的错误

二、Class的获取

反射的第一步需要获取Class对象,怎么获取呢

首先构建一个pojo对象Person作为例子

@Slf4jpublic class Person {static {log.info("I am person");}public Person(String name){this.name=name;}public Person(){}public int id;private String name;private int age;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}@Mgpublic void showMessage(){log.info("this person message");}}

1、实例化对象获取Class对象

Person person = new Person();Class cl1 = person.getClass();

2、类获取Class对象

Class cl2 = Person.class;

3、使用class.forName根据类的全名获取

Class cl3 = Class.forName("com.mg.empty.demo.mg20200531.Person");

4、使用类加载器获取

Class cl4 = ClassLoader.getSystemClassLoader().loadClass("com.mg.empty.demo.mg20200531.Person");

附:forName和loadClass的区别

类的加载过程包括以下步骤:加载、连接(连接过程暂不细分)、初始化

forName默认是进行到初始化,loadClass进行到连接这一步

即,loadClass只把类加载到了jvm中,forName加载并完成了初始化

这个怎么证明呢,在Person这个类中有一块代码

static {System.out.println("i am person");}

这个是在初始化阶段执行的,试下两个方法的日志打印有什么不同

如果forName不想走初始化过程,可以使用public static Class> forName(String name, boolean initialize, ClassLoader loader) 这个方法加载,将initialize设置为false即可

三、反射的使用

1、构造函数的获取和使用

首先获取构造函数,代码如下

Class cl = Person.class;String pStr = "";log.info("------------获取所有构造函数----------------");Constructor[] cs = cl.getConstructors();for (Constructor c : cs) {Parameter[] ps = c.getParameters();pStr = "";for (Parameter p : ps) {pStr += p.getType().getSimpleName() + " " + p.getName() + ",";}log.info(String.format("%s (%s)", c.getName(), pStr));}

输出结果为

com.mg.empty.demo. Person (String name,)com.mg.empty.demo. Person ()

获取指定构造函数

log.info("------------获取指定构造函数----------------");Constructor sc = cl.getConstructor(String.class);Parameter[] ps = sc.getParameters();pStr = "";for (Parameter p : ps) {pStr += p.getType().getSimpleName() + " " + p.getName() + ",";}log.info(String.format("%s (%s)", sc.getName(), pStr));

使用构造函数实例化对象

log.info("------------使用构造函数----------------");Constructor sc1 = cl.getConstructor(String.class);Person person1 = (Person) sc1.newInstance("mg");log.info(person1.getName());

2、属性的获取和使用

获取属性的方法有两个getFields只能获取public修饰的属性,如下

log.info("------------获取public属性----------------");Field[] fs = cl.getFields();for (Field f : fs) {log.info(String.format("%s %s", f.getType().getSimpleName(), f.getName()));}

获取所有属性的方法是getDeclaredFields

log.info("------------获取所有属性----------------");Field[] fs = cl.getDeclaredFields();for(Field f : fs){log.info(String.format("%s %s",f.getType().getSimpleName(),f.getName()));}

获取指定名称的属性

log.info("------------获取指定属性----------------");Field field = cl.getDeclaredField("name");log.info(String.format("%s %s",field.getType().getSimpleName(),field.getName()));

获取和修改属性值

log.info("------------获取和修改属性值----------------");Person person1 = new Person();person1.setName("mg");Field field = cl.getDeclaredField("name");field.setAccessible(true);log.info(field.get(person1).toString());field.set(person1,"mg2020");log.info(field.get(person1).toString());

注意 private的属性需要设置下field.setAccessible(true);否则抛异常。

3、方法的获取和使用

获取包括父类的所有方法

log.info("------------获取包括父类的所有方法----------------");Method[] ms = cl.getMethods();for (Method m : ms) {Parameter[] ps = m.getParameters();pStr = "";for (Parameter p : ps) {pStr += p.getType().getSimpleName() + " " + p.getName() + ",";}log.info(String.format("%s (%s)", m.getName(), pStr));}

获取本类中定义的方法

log.info("------------获取所有方法----------------");Method[] ms = cl.getDeclaredMethods();for (Method m : ms) {Parameter[] ps = m.getParameters();pStr = "";for (Parameter p : ps) {pStr += p.getType().getSimpleName() + " " + p.getName() + ",";}log.info(String.format("%s (%s)", m.getName(), pStr));}

获取指定的方法

log.info("------------获取指定方法----------------");Method method = cl.getMethod("setName",String.class);Parameter[] ps = method.getParameters();pStr = "";for (Parameter p : ps) {pStr += p.getType().getSimpleName() + " " + p.getName() + ",";}log.info(String.format("%s (%s)", method.getName(), pStr));

使用方法

{log.info("------------使用方法----------------");Person person1 = new Person();Method method = cl.getMethod("setName",String.class);method.invoke(person1,"mg");log.info(person1.getName());}

4、注解

先定义一个注解

@Retention(RetentionPolicy.RUNTIME)@Target(value={ElementType.METHOD})public @interface Mg {String name() default "mg";}

接下来获取注解属性

log.info("------------获取注解----------------");Method method = cl.getDeclaredMethod("showMessage");Mg annotation = method.getAnnotation(Mg.class);log.info(annotation.name());

反射到这里就结束了,虽然反射是一个很简单的技术,并且日常编码中不经常使用,不过在框架中是很重要的存在,所以还是需要熟练掌握的。

文中代码提取 关注公众号 MG驿站 输入 泛型 获取

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
标题:深入理解Java反射机制 摘要:Java反射机制是一种强大的特性,它允许在运行时动态地获取和操作类的信息。本文将深入探讨Java反射机制的原理、应用场景以及使用方法。 正文: 一、什么是Java反射机制? Java反射机制是指在运行时通过程序获取类的信息,并且可以动态调用类的方法、访问和修改类的属性,以及创建对象等操作。它使得我们可以在不知道类的具体信息的情况下,对类进行操作。 二、Java反射机制的原理 Java反射机制的核心类是Class类,它代表一个类或接口。通过Class类,我们可以获取类的构造方法、方法、属性等信息。Java反射机制主要涉及以下几个重要的类和接口: 1. Class类:代表一个类或接口,在运行时可以获取类的相关信息。 2. Constructor类:代表类的构造方法,可以用来创建对象。 3. Method类:代表类的方法,可以用来调用方法。 4. Field类:代表类的属性,可以用来访问和修改属性的值。 通过这些核心类和接口,我们可以在运行时动态地加载和使用类,而不需要在编译时确定类的具体信息。 三、Java反射机制的应用场景 Java反射机制在许多框架和工具中得到广泛应用,下面列举几个常见的应用场景: 1. 框架开发:许多框架,如Spring、Hibernate等,都使用了反射机制来实现动态创建对象、调用方法等功能。 2. 单元测试:JUnit等测试框架使用反射机制来执行被测试类的方法。 3. 配置文件解析:通过反射机制,可以根据配置文件中的类名动态加载和初始化对象。 4. 动态代理:反射机制可以实现动态代理,通过在运行时生成代理对象,来增强方法的功能。 四、Java反射机制的使用方法 Java反射机制提供了一系列的方法来操作类,包括获取类名、获取构造方法、调用方法、访问和修改属性等。下面是一些常用的反射方法: 1. 获取类名:使用Class类的getName()方法可以获取类的全限定名。 2. 获取构造方法:使用Class类的getConstructors()或getDeclaredConstructors()方法可以获取类的构造方法。 3. 调用方法:使用Method类的invoke()方法可以调用类的方法。 4. 访问和修改属性:使用Field类的get()和set()方法可以访问和修改类的属性。 需要注意的是,在使用反射机制时,要处理好异常情况,并且在性能要求较高的场景下,应谨慎使用反射,以避免性能开销过大的问题。 结论: Java反射机制是一种强大的特性,它允许在运行时动态地获取和操作类的信息。通过反射机制,我们可以实现诸如动态创建对象、调用方法、访问和修改属性等功能。尽管反射机制具有一定的性能开销,但在许多框架和工具中得到广泛应用。了解和掌握Java反射机制,将会为我们的开发工作带来更多的灵活性和便利性。 参考文献: - Oracle官方文档 - Java反射:https://docs.oracle.com/javase/tutorial/reflect/ - 廖雪峰的Java教程 - 反射:https://www.liaoxuefeng.com/wiki/1252599548343744/1264804593397984

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值