18

我对 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);
    }
}

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

那么如何实现另一个呢?

4

3 回答 3

37

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

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

那么让我们看一下可执行代码:

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

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

委托示例

委托由对象在内部完成 - 与调用方法的方式无关。例如,如果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 something部分getHelp()。当使用回调时,回调不做任何与对象行为相关的事情:它只是以某种方式通知调用者,这就是为什么// 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
        }
    }
于 2012-09-29T18:12:04.050 回答
6

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

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

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

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

于 2012-09-29T17:58:17.257 回答
1

您想要实现的是原始调用者和服务之间的双向通信,同时避免服务依赖于客户端。您用于该目标的模式通常取决于您的语言限制。您使用函数指针、闭包,或者,如果您没有这些,则使用回调对象(也可能被视为闭包)。

然后,对于相同或非常相似的模式,通常会有很多不同的名称。

于 2012-09-29T17:58:45.057 回答