java基础——反射与回调
一、java的反射
java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取信息以及动态
调用对象的方法的功能成为java语言的反射机制。
java的反射机制主要提供了以下功能:
• 在运行时判断任意一个对象所属的类;
• 在运行时构造任意一个类的对象;
• 在运行时判断任意一个类所具有的成员变量和方法;
• 在运行时调用任意一个对象的方法;
• 生成动态代理。
1、JVM创建Class类对象的过程
2、获取class的三种方式
- Class.forName(“com.weige.Test”);
- Test.class;
- new Test().getClass();
3、反射的作用
Java中编译类型有两种
静态编译:在编译时确定类型,绑定对象即通过
动态编译:运行时确定类型,绑定对象
反射允许静态语言在运行时检查、修改程序的结构与行为。 在静态语言中,使用一个变量时,必须知道它的类型;在Java中,变量的类型信息在编译时都保存到了class文件中,这样在运行时才能保证准确无误。换句话说,程序在运行时的行为都是固定的。如果想在运行时改变,就需要Java的反射机制。
动态编译最大限度地发挥了Java的灵活性,体现了多态的应用,可以减低类之间的耦合性。 Java反射是Java被视为动态(或准动态)语言的一个关键性质。
4、反射的应用场景
- 最重要的用途就是开发各种通用框架,注入属性,调用方法,如Spring
- JDBC中,利用反射动态加载了数据库驱动程序
- Web服务器中利用反射调用了Sevlet的服务方法
- Eclispe等开发工具利用反射动态刨析对象的类型与结构,动态提示对象的属性和方法
5、反射机制的优缺点?
- 优点:可以动态执行,在运行期间根据业务功能动态执行方法、访问属性,最大限度发挥了java的灵活性。
- 缺点:对性能有影响,这类操作总是慢于直接执行java代码。
A、反射实例——通过属性调用对象
public static void main(String[] args) throws Exception{ Class<?> a = new Test().getClass(); Test test = new Test("456"); Field field= a.getDeclaredField("age"); field.set(test, "123"); System.out.println(field.get(test)); }
B、反射实例——通过方法调用对象
public static void main(String[] args) throws Exception{ Class<?> a = new Test().getClass(); Method method = a.getDeclaredMethod("say", String.class,int.class); method.invoke(a.newInstance(), "asd",1); } private static void say(String a,int b) { System.out.println("测试"+a+b); }
二、函数的回调
在Java里面,我们使用接口来实现回调。回调函数,或简称回调,是指通过函数参数传递到其它代码的,某一块可执行代码的引用。这一设计允许了底层代码调用在高层定义的子程序。
程序员A写了一段程序(程序a),其中预留有回调函数接口,并封装好了该程序。程序员B要让a调用自己的程序b中的一个方法,于是,他通过a中的接口回调自己b中的方法。目的达到。在C/C++中,要用回调函数,被掉函数需要告诉调用者自己的指针地址,但在JAVA中没有指针,怎么办?我们可以通过接口(interface)来实现定义回调函数。
假设我是程序员A,以下是我的程序a:
public class Caller { public MyCallInterface mc; public void setCallfuc(MyCallInterface mc) { this.mc= mc; } public void call(){ this.mc.method(); } }
我还需要定义一个接口,以便程序员B根据我的定义编写程序实现接口。
public interface MyCallInterface { public void method(); }
于是,程序员B只需要实现这个接口就能达到回调的目的了:
public class B implements MyCallInterface { public void method() { System.out.println("回调"); } public static void main(String args[]) { Caller call = new Caller(); call.setCallfuc(new B()); call.call(); } }
主要就是回调者需要实现一个回调接口,然后把自己,也就是相当于接口当做参数传给别人,别人拿到这个接口之后,最终都会调用接口中的方法,就相当于执行了回调方法。
下面举个通俗的例子:
某天,我打电话向你请教问题,当然是个难题你一时想不出解决方法,我又不能拿着电话在那里傻等,于是我们约定:等你想出办法后打手机通知我,这样,我就挂掉电话办其它事情去了。过了XX分钟,我的手机响了,你兴高采烈的说问题已经搞定,应该如此这般处理。故事到此结束。这个例子说明了“异步+回调”的编程模式。其中,你后来打手机告诉我结果便是一个“回调”过程;我的手机号码必须在以前告诉你,这便是注册回调函数;我的手机号码应该有效并且手机能够接收到你的呼叫,这是回调函数必须符合接口规范。
参考:
https://blog.csdn.net/qq_30346471/article/details/84299993
https://blog.csdn.net/weixin_45864705/article/details/126271798