首页 > 基础资料 博客日记

【Java】接口回调和方法回调的简单教程

2025-01-14 13:00:09基础资料围观50

本篇文章分享【Java】接口回调和方法回调的简单教程,对你有帮助的话记得收藏一下,看Java资料网收获更多编程知识

先来聊接口回调

接口回调

这是一种设计模式,我的理解:

共有三者:调用者,被调用者和接口方法

接口回调的核心在于:调用者只需要实现接口方法,而被调用者只需要在合适的时机反过来调用调用者实现的方法,通过第三者接口方法,实现了双向解耦,调用者不关心被调用者,被调用者不关心调用者如何实现方法的。

回调的核心思想是:

  1. 调用者(Caller) 提供一个实现,交给 被调用者(Callee)
  2. 被调用者在适当的时机,通过调用该实现来执行相应的逻辑。

"回调"的体现在于:正常都是调用者调用被调用者,而这里是被调用者主动调用调用者


接口回调的实现步骤

1. 定义接口

  • 定义一个接口,其中包含需要被回调的方法。

2. 实现接口

  • 由调用者(Caller)实现这个接口,并提供具体的回调逻辑。

3. 注册接口

  • 将实现接口的对象传递给被调用者(Callee)。

4. 在合适时机回调

  • 被调用者在需要时调用接口中的方法,触发回调逻辑。

举例理解

举一个生活中的例子,比如点外卖

这涉及到三者关系:

  • app用户(调用者)
  • 饿了么app(被调用者)
  • 联系方式(接口方法实现)
  • 【用户】填写了联系方式,比如地址和手机号(实现接口方法)
  • 付款后,【饿了么】开始一系列操作(呼叫商家 ->呼叫骑手等)
  • 无论如何,最终送餐的时候必须知道如何联系【用户】,会调用实现的接口方法
  • 最终送到【用户】手中

这个例子中,想想看,【用户】作为调用者需要关心【饿了么】的一系列流程操作吗?并不需要。【饿了么】需要关心【用户】的联系方式吗?呃。。。。好像的确挺关心的,这个例子翻车了?反正你能理解就行,联系方式就算写错成其他户的地址,【饿了么】的骑手也会照样送过去,这么看也不算关心内容了。

代码案例

写一个简单的代码案例,加深理解

public interface ButtonClickListener {
    void click(); // 回调方法
}

public class Button {
    private ButtonClickListener listener;
    public void setButtonClickListener(ButtonClickListener listener){
        this.listener = listener;
    }
    public void click(){
        if (listener != null){
            listener.click(); // 回调接口方法
        }
    }
}

public class Main implements ButtonClickListener{
    public static void main(String[] args) {
        // 创建被调用者实例
        Button button = new Button();
        // 创建调用者(Main)实例
        Main main = new Main();
        // 注册回调接口
        button.setButtonClickListener(main);
        // 模拟按钮点击
        button.click();
    }
    @Override
    public void click() {
        System.out.println("点击按钮");
    }
}

常见的接口回调

举一个常用的Consumer接口经常出现接口回调现象

关于Consumer接口可以看我另一篇博客:【Java】Java8的4个函数式接口简单教程-CSDN博客

Arrays.asList("a", "b", "c")
                .forEach(s -> System.out.println(s));

如上,这行代码其实发生了隐式的接口回调

三者关系:

  • 我(调用者):我调用了集合的forEach()方法,并且我实现了Consumer接口的抽象方法accept()
  • forEach()(被调用者):forEach()方法会自动调用我实现的accept()方法
  • Consumer.accept()(接口方法)

我只需要实现接口方法,forEach()会主动调用我实现的方法,并不关心我到底咋实现的

异步回调

上面说的都是同步回调

  • 回调方法在调用的过程中立即被执行。
  • 示例:按钮点击监听。

异步回调是:

  • 回调方法在调用结束后,异步地被执行(通常是在另一个线程中)。
  • 示例:网络请求完成后通知调用者。
// 异步回调示例(模拟异步任务)
public class AsyncTask {

    // 定义回调接口
    public interface TaskCallback {
        void onTaskComplete(String result);
    }

    // 执行异步任务
    public void execute(TaskCallback callback) {
        new Thread(() -> {
            try {
                Thread.sleep(2000);  // 模拟耗时任务
                String result = "Task Finished!";
                callback.onTaskComplete(result);  // 回调通知任务完成
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }).start();
    }
}

// 调用者代码
public class Main {
    public static void main(String[] args) {
        AsyncTask task = new AsyncTask();
        task.execute(result -> System.out.println("Callback received: " + result));
        System.out.println("Task started...");
    }
}

注意事项

空指针检查

  • 如果没有设置回调接口(即 listenernull),直接调用接口方法会抛出 NullPointerException
if (listener != null) {
    listener.onClick();
}

线程安全

  • 在多线程场景下,注意回调接口的线程安全问题,例如异步任务回调需要在主线程中执行。

性能开销

  • 回调接口的调用会带来一定的性能开销,尤其是在高频调用场景中,可能对性能产生影响。

强依赖问题

  • 如果接口实现的方式过于复杂,会导致代码难以维护,因此需要设计合理的接口和实现。

方法回调

差不多,看区别就行

核心记住一句话:将方法作为另一个方法的参数

方法回调:

  • 调用者将某个对象的方法引用直接传递给被调用者,被调用者在适当时机调用这个方法。
  • 不依赖接口,通常通过方法引用或 Lambda 表达式实现。
依赖接口不需要接口,直接传递方法引用需要接口,调用者实现接口并提供逻辑
灵活性不灵活,直接绑定到具体类的方法更灵活,支持不同实现类
解耦程度较低,调用者和被调用者直接关联高度解耦,通过接口隔离调用者与实现者
复杂性简单相对复杂,但更适合扩展场景
  • 方法回调:简单直接调用具体类的方法,适用于调用单一方法的场景。
  • 接口回调:强调解耦,适用于需要灵活多变的回调逻辑场景。

总结就行,区别不大,就是不用实现接口了

三者关系就变成,我调用forEach(),而forEach()调用了我写的Caller.printMessage()方法


文章来源:https://blog.csdn.net/lklalmq/article/details/144570563
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!

标签:

相关文章

本站推荐

标签云