По словам Тома Хотина
Замыкание - это блок кода, на который можно ссылаться (и передавать) с доступом к переменным охватывающей области.
Теперь я пытаюсь имитировать пример закрытия JavaScript в Википедии с « прямым » переводом на Java, в надежде быть полезным:
//ECMAScript
var f, g;
function foo() {
var x = 0;
f = function() { return ++x; };
g = function() { return --x; };
x = 1;
print('inside foo, call to f(): ' + f()); // "2"
}
foo();
print('call to g(): ' + g()); // "1"
print('call to f(): ' + f()); // "2"
Теперь часть java: Function1 - это интерфейс «Functor» с арностью 1 (один аргумент). Замыкание - это класс, реализующий Function1, конкретный Functor, который действует как функция (int -> int). В методе main () я просто создаю экземпляр foo как объект Closure, реплицируя вызовы из примера JavaScript. Класс IntBox - это простой контейнер, он ведет себя как массив из 1 int:
int a [1] = {0}
interface Function1 {
public final IntBag value = new IntBag();
public int apply();
}
class Closure implements Function1 {
private IntBag x = value;
Function1 f;
Function1 g;
@Override
public int apply() {
// print('inside foo, call to f(): ' + f()); // "2"
// inside apply, call to f.apply()
System.out.println("inside foo, call to f.apply(): " + f.apply());
return 0;
}
public Closure() {
f = new Function1() {
@Override
public int apply() {
x.add(1);
return x.get();
}
};
g = new Function1() {
@Override
public int apply() {
x.add(-1);
return x.get();
}
};
// x = 1;
x.set(1);
}
}
public class ClosureTest {
public static void main(String[] args) {
// foo()
Closure foo = new Closure();
foo.apply();
// print('call to g(): ' + g()); // "1"
System.out.println("call to foo.g.apply(): " + foo.g.apply());
// print('call to f(): ' + f()); // "2"
System.out.println("call to foo.f.apply(): " + foo.f.apply());
}
}
Он печатает:
inside foo, call to f.apply(): 2
call to foo.g.apply(): 1
call to foo.f.apply(): 2