闭包(closure)是一个可调用的对象,它记录了一些信息,这些信息来自于创建它的作用域。通过这个定义,可以看出内部类是面向对象的闭包,因为它不仅包含外围类对象(创建内部类的作用域)的信息,还自动拥有一个指向此外围类对象的引用,在此作用域内,内部类有权操作所有的成员,包括private成员。
Java最引人争议的问题之一就是,人们认为Java应该包含某种类似指针的机制,以允许回调(callback)。通过回调,对象能够携带一些信息,这些信息允许它在稍后的某个时刻调用初始的对象。
如果回调是通过指针实现的,那么就只能寄希望于程序员不会误用该指针。然而,您应该已经了解到,Java更小心仔细,所以没有在语言中包括指针。
对于Java而言,通过内部类提供闭包的功能是优良的解决方案,它比指针更灵活、更安全。
// Using inner classes for callbacks
package innerclasses
import static net.mindview.util.Print.*;
interface Incrementable {
void increment();
}
// Very simple to just implement the interface
class Callee1 implements Incrementable {
private int i = 0;
public void increment() {
i++
print(i);
}
}
class MyIncrement {
public void increment() {print("Other operation");}
static void f(MyIncrement mi) {mi.increment();}
}
// If your class must implement increment() in
// some other way, you must use an inner class:
class Callee2 extends MyIncrement {
private int i=0;
public void increment() {
super.increment();
i++;
print(i);
}
private class Closure implements Incrementable {
public void increment() {
// Specify outer-class method, otherwise
// you'd get an infinite recursion
Callee2.this.increment();
}
}
Incrementable getCallbackReference() {
return new Closure();
}
}
class Caller {
private Incrementable callbackReference;
Caller(Incrementable cbh) {callbackReference = cbh;}
void go() {callbackReference.increment();}
}
public class Callbacks {
public static void main(String[] args) {
Callee1 c1 = new Calleel();
Callee2 c2 = new Callee2();
MyIncrement.f(c2);
Caller caller1 = new Caller(c1);
Caller caller2 = new Caller(c2.getCallbackReference());
caller1.go();
caller1.go();
caller2.go();
caller2.go();
}
}
输出结果为
Other operation
1
1
2
Other operation
2
Other operation
3
内部类Closure实现了Incrementable,以提供一个返Callee2的“钩子”(hook)——而且是一个安全的钩子。无论谁获得此 Incrementable的引用,都只能调用increment(),除此之外没有其他功能(不像指针那样,允许您做很多事情)。
Caller的构造器需要一个Incrementable的引用作为参数(虽然可以在任意时刻捕获回调引用),然后在以后的某个时刻,Caller对象可以使用此引用回调Callee类。
回调的价值在于它的灵活性——可以在运行时动态地决定需要调用什么方法。
分享到:
相关推荐
补充说明:闭包可以使用USE关键连接外部变量。 总结:PHP闭包的特性...不过匿名函数还是挺有用的,比如在使用preg_replace_callback等之类的函数可以不用在外部声明回调函数了。合理使用闭包能使代码更加简洁和精炼。
将`Callable`和`Closure`结合使用,可以确保我们的代码更加健壮。例如,如果我们需要一个接收回调函数的方法,我们可以这样做: ```php function processWithCallback(callable $callback) { // ...执行一些操作.....
PHP中的闭包(Closure)匿名函数是在PHP 5.3版本引入的新特性,它是一种能够捕获和操作外部变量的函数。闭包提供了一种在函数内部访问和修改外部作用域变量的能力,而无需通过全局变量或类成员来实现。这在很多场景...
* @param \Closure $callback * @return void */ public function __construct($callback) { $this->callback = $callback; }我们知道,在使用语言的过程中,肯定会关注在实际开发过程...
闭包的语法很简单,需要注意的关键字就只有use,use是连接闭包和外界变量。 复制代码 代码如下: $a = function() use($b) {} 简单例子如下: 复制代码 代码如下: function callback($fun) { $fun(); } $msg = ...
在Node.js Little Book中文版中,作者首先介绍了Node.js的基本概念,包括Event Loop、Scope与Closure、Callback、CPS(Continuation-Passing Style)等。这些概念是Node.js开发的基础,理解这些概念对于后续的学习和...
使用 Closure 可以避免全局变量的使用,提高了代码的可读性和维护性,同时也可以简洁地实现回调函数的参数传递。 以下是一个使用 Closure 的实例代码: ``` var callback = { success: function(data) { var item...
Swift提供了多种简化闭包语法的方式,如自动类型推断、单表达式闭包和trailing closure语法。例如,下面的`map`调用可以更简洁地表示: ```swift let squaredNumbers = numbers.map({ $0 * $0 }) // 等价于 map...
- `Cache::remember($key, $minutes, Closure $callback)`:如果缓存不存在,则运行回调函数并存储结果,返回缓存或新值。 **四、缓存标签** 缓存标签允许你在同一缓存存储中组织多个相关值。通过标签,可以同时...
通常,我们可以通过代理(Delegate)、闭包(Closure)或者通知(Notification)来实现,但block提供了一种更为简洁和灵活的方法。以下是如何使用block实现反向传值的步骤: 1. **创建Block属性**:在A界面的视图...
闭包(closure)是函数式编程中的概念,出现于 20 世纪 60 年代,最早实现闭包的语言是 Scheme,它是 LISP 的一种方言。之后闭包特性被其他语言广泛吸纳。 闭包的严格定义是“由函数(环境)及其封闭的自由变量组成...
* 闭包(closure)是指函数可以访问其外层作用域的变量和函数。 四、this 关键字 * this 关键字是 JavaScript 中的一个特殊关键字,用于指代当前对象。 * this 关键字的含义可以根据不同的上下文而变化。 五、...
这可能是通过使用其他函数,如`preg_replace`配合匿名函数(`Closure`)或者使用`preg_match_all`结合数组处理来实现的。 标签“ecmall”指的是一个开源的电子商务平台,它主要用于构建B2C类型的在线商店。Ecmall...
closure_t *cl = closure_new((void (*)(closure_t *, int))my_callback, &data); // 调用闭包 closure_call(cl, 10); // 释放资源 closure_free(cl); return 0; } ``` 以上代码展示了如何创建一个简单的...
2.2 Scope 與 Closure . . . . . . . . . . . . . . . . . . . . . . . . . . 12 2.3 Callback . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13 2.4 CPS(Continuation-Passing Style) . . . ...
iOS的GCD(Grand Central Dispatch)和Android的AsyncTask或Retrofit的Callback接口可以实现这一目的。 7. **状态管理**:为了给用户提供良好的反馈,开发者需要管理各种状态,如“刷新中”、“加载中”、“无更多...
5. **预定义类**:增加了几个预定义类,如 `Closure` 和 `Reflection`,便于操作闭包和反射,增强元编程能力。 **新函数与改进** 1. **`void` 返回类型**:引入了 `void` 关键字作为函数返回类型,表示函数没有...