Live2D

java基础——反射与回调

一、java的反射

   java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取信息以及动态

   调用对象的方法的功能成为java语言的反射机制。

   java的反射机制主要提供了以下功能:

    • 在运行时判断任意一个对象所属的类;

    • 在运行时构造任意一个类的对象;

    • 在运行时判断任意一个类所具有的成员变量和方法;

       • 在运行时调用任意一个对象的方法;

    • 生成动态代理。

  1、JVM创建Class类对象的过程

                                         

  2、获取class的三种方式

  1. Class.forName(“com.weige.Test”);
  2. Test.class;
  3. 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

posted @ 2023-04-07 15:09  -涂涂-  阅读(244)  评论(0编辑  收藏  举报