Java 中的委托与回调委派示例

2022-09-02 19:45:47

我对Java中的委托和回调术语有一些误解。

class MyDriver {

    public static void main(String[] argv){
        MyObject myObj = new MyObject();
        // definition of HelpCallback omitted for brevity
        myObj.getHelp(new HelpCallback () {
            @Override
            public void call(int result) {
                System.out.println("Help Callback: "+result);
            }
        });
    }
}

class MyObject {

    public void getHelp(HelpCallback callback){
        //do something
        callback.call(OK);
    }
}

是回调还是委托(委托和回调是相同还是相似?)?

如何实现另一个?


答案 1

这是一个回调。根据维基百科

在计算机编程中,回调是对作为参数传递给其他代码的一段可执行代码的引用。

因此,让我们看一下可执行代码:

public void getHelp(HelpCallback callback){
    //do something
    callback.call(OK);
}

此处,参数是对 类型对象的引用。由于该引用是作为参数传入的,因此它是一个回调。callbackHelpCallback

委派示例

委派由对象在内部完成 - 与调用方法的方式无关。例如,如果变量不是参数,而是实例变量:callback

class MyDriver {

    public static void main(String[] argv){
        // definition of HelpStrategy omitted for brevity
        MyObject myObj = new MyObject(new HelpStrategy() {
            @Override
            public void getHelp() {
                System.out.println("Getting help!");
            }
        });
        myObj.getHelp();
    }

}

class MyObject {
    private final HelpStrategy helpStrategy;

    public MyObject(HelpStrategy helpStrategy) {
        this.helpStrategy = helpStrategy;
    }

    public void getHelp(){
        helpStrategy.getHelp();
    }
}

...那么它将是授权。

此处,使用策略模式。有两点需要注意:MyObject

  1. 调用 不涉及传递对可执行代码的引用。即,这不是回调。getHelp()
  2. 从对象的公共接口或调用中看不到调用的事实。这种信息隐藏在某种程度上是委派的特征。MyObject.getHelp()helpStrategy.getHelp()MyObjectgetHelp()

同样值得注意的是,该方法中缺少一节。使用回调时,回调不会执行与对象行为相关的任何操作:它只是以某种方式通知调用方,这就是为什么需要一个部分的原因。但是,当使用委托时,该方法的实际行为取决于委托 - 因此我们实际上可能最终需要两者,因为它们具有不同的目的:// do somethinggetHelp()// do something

    public void getHelp(HelpCallback callback){
        helpStrategy.getHelp(); // perform logic / behavior; "do something" as some might say
        if(callback != null) {
            callback.call(); // invoke the callback, to notify the caller of something
        }
    }

答案 2

我认为“回调”是通用模式的名称,您可以在其中为要调用的模块提供一种使所述模块调用代码的方法。C# 委托或 ObjC 委托对象(这两者是完全不同的野兽)或 Java 类实现回调接口是不同的、特定于平台的实现回调模式的方式。(它们本身也可以被视为模式。其他语言或多或少有不同的方式。

上述“委托”的概念也类似于策略模式,在策略模式中,委托可以被视为一个。同样,访问者也是一种回调类型。(访问者也是处理每个访问项目的策略。

所有这些都使用了对我来说很直观的定义,对其他人来说可能不是,因为“回调”或“委托”都不是正式术语,如果不参考在你的上下文中有效的先前定义,讨论它们就没有意义。因此,问这个定义是什么是没有意义的,因为据我所知,没有一个权威的定义。也就是说,这个问题的其他答案可能会说一些完全不同的东西。

我的建议是专注于你的设计的优点 - 它是否实现了你需要的东西,不引入紧密耦合等 - 而不是语义的细节。当两种设计模式看起来相似时,它们可能可以用来同样好地实现相似的目标。


推荐