De acordo com Tom Hawtin
Um fechamento é um bloco de código que pode ser referenciado (e transmitido) com acesso às variáveis do escopo envolvente.
Agora estou tentando emular o exemplo de fechamento de JavaScript na Wikipedia , com uma tradução " direta " para Java, na esperança de ser útil:
//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"
Agora a parte java: Function1 é a interface "Functor" com arity 1 (um argumento). Closure é a classe que implementa a Function1, um Functor concreto que atua como função (int -> int). No método main (), apenas instanciarei foo como um objeto Closure, replicando as chamadas do exemplo de JavaScript. A classe IntBox é apenas um contêiner simples, ela se comporta como uma matriz de 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());
}
}
Ele imprime:
inside foo, call to f.apply(): 2
call to foo.g.apply(): 1
call to foo.f.apply(): 2