This article explains Java lambdas—concise anonymous functions—and their use with functional interfaces (interfaces with one abstract method). It details lambda syntax, contrasts them with anonymous inner classes highlighting brevity and readability
Java lambdas are concise, anonymous functions that provide a functional programming paradigm within Java. They're essentially blocks of code that can be passed around as arguments to methods or stored in variables. To use them, you need a functional interface, which is an interface with exactly one abstract method (although it can have multiple default methods or static methods). The lambda expression's signature implicitly matches the functional interface's abstract method.
Here's a simple example using the Runnable
interface (a built-in functional interface):
Runnable myRunnable = () -> System.out.println("Hello from a lambda!"); new Thread(myRunnable).start();
This code creates a lambda expression () -> System.out.println("Hello from a lambda!");
that implements the run()
method of the Runnable
interface. The empty parentheses ()
indicate that the lambda takes no arguments. The arrow ->
separates the parameter list from the body.
Another example using a lambda with parameters:
interface StringOperation { String operate(String str); } StringOperation reverseString = (str) -> new StringBuilder(str).reverse().toString(); String reversed = reverseString.operate("hello"); // reversed will be "olleh"
Here, StringOperation
is a functional interface. The lambda expression (str) -> new StringBuilder(str).reverse().toString()
takes a String
argument (str
) and returns a reversed String
.
Anonymous inner classes, while achieving similar functionality, are significantly more verbose than lambdas. Using lambdas offers several key advantages:
Consider the Runnable
example again. The anonymous inner class equivalent would be:
Runnable myRunnable = new Runnable() { @Override public void run() { System.out.println("Hello from an anonymous inner class!"); } }; new Thread(myRunnable).start();
This is clearly longer and less expressive than the lambda version.
Creating your own functional interface is straightforward. Simply define an interface with exactly one abstract method. You can add default methods and static methods as needed. The @FunctionalInterface
annotation is optional but recommended. It helps the compiler enforce the single abstract method rule and catch errors early.
@FunctionalInterface interface MyFunctionalInterface { int calculate(int a, int b); default int add(int a, int b){ return a b; } } // Usage MyFunctionalInterface myOperation = (a, b) -> a * b; int result = myOperation.calculate(5, 3); // result will be 15 int sum = myOperation.add(5,3); // result will be 8
In this example, MyFunctionalInterface
is a functional interface with one abstract method calculate()
. The @FunctionalInterface
annotation indicates this. The add()
method is a default method.
Java lambdas are widely supported by many existing Java libraries and frameworks. They're integrated into core Java APIs and are commonly used with:
For example, using lambdas with the Streams API:
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5); int sum = numbers.stream().map(n -> n * 2).sum(); // sum will be 30
This code uses a lambda n -> n * 2
within the stream pipeline to double each number before summing them. This is a much cleaner approach compared to using traditional iteration. The widespread adoption of lambdas makes them an essential part of modern Java development.
The above is the detailed content of How do I use Java lambdas and functional interfaces?. For more information, please follow other related articles on the PHP Chinese website!