Ich verstehe nicht wozu man funktionale interfaces braucht? Was genau ist der Sinn? und kann man funktionale interfaces instanziieren?
(int a, int b) -> a + b
.void actionPerformed(ActionEvent e);
public void addActionListener(ActionListener l)
button.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
// mach was (ggf. mit e)
}
});
button.addActionListener(e -> System.out.println("Pressed " + e));
ActionListener listener = e -> System.out.println("Pressed " + e);
button.addActionListener(listener);
Vielen dank für die ausführliche Erklärung!Das ist relativ einfach: ein funktionales Interface dient dazu, eine Funktion zu deklarieren. Die Unterscheidung, ob ein Interface funktional ist oder nicht, besteht einfach darin, wie viele nicht definierte Methoden das Interface enthält: falls nur eine Methode nicht definiert ist, handelt es sich um ein funktionales Interface, sonst nicht.
Benutzt wird das ganze im Zusammenhang mit Lambdas, also anonymen Funktionen, um eine Implementierung dieses Interfaces (sprich: die Funktion) anzugeben.
Nehmen wir mal eine Funktion an, die zwei Ganzzahlen addiert. Mathematisch formal ließe sich das schreiben als f:int x int -> int mit f(a, b) := a + b. Das ist jetzt aber eine benannte Funktion f. Lassen wir den Namen weg, erhalten wir int x int -> int mit (a,b) := a+b oder kürzer (int a, int b) := a + b. Jetzt sind wir schon fast bei der Lambda-Schreibweise von Java. Dort würde er Lambda-Ausdruck so aussehen(int a, int b) -> a + b
.
Wozu braucht man jetzt das Interface? Nun, Java ist eine Sprache mit statischer Typsicherheit. Zur Übersetzungszeit müssen die Typen also bekannt sein. Das Problem ist also, wie man z. B. einen Parameter angibt, der eine Funktion entgegennimmt, die dann auch noch anonym implementiert werden kann. Die Lösung ist recht einfach: ein funktionales Interface.
Nehmen wir mal java.awt.event.ActionListener. Dieses Interface bestitzt genau eine Methode (=Funktion):
und ist damit ein funktionales Interface.Java:void actionPerformed(ActionEvent e);
Die Methode addActionListener von z. B. JButton erwartet nun ein Objekt mit ActionListener-Schnittstelle.
Java:public void addActionListener(ActionListener l)
Statt nun das Interface mit Hilfe einer anonymen Klasse zu implementieren
kann an jetzt auf einen Lambda-Ausdruck zurückgreifen:Java:button.addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { // mach was (ggf. mit e) } });
Natürlich kann man das ActionListener-Objekt auch explizit definieren:Java:button.addActionListener(e -> System.out.println("Pressed " + e));
Java:ActionListener listener = e -> System.out.println("Pressed " + e); button.addActionListener(listener);