Java 8 Functional Interface Examples

In Java 8, a functional interface is an interface that has exactly one abstract method. It is also known as a Single Abstract Method (SAM) interface.

Java 8 introduced functional interfaces to enable the use of lambda expressions, which provide a more concise and expressive way to write functional-style code.

To mark an interface as a functional interface, we can use the @FunctionalInterface annotation. While using the annotation is optional, it’s a good practice to add it as it helps to avoid the accidental addition of more abstract methods to the interface, thus preserving its functional nature.

Note – Without a functional interface we can’t use a lambda expression.

Important points related to the functional interface.

  1. Single Abstract Method (SAM) : A functional interface must have exactly one abstract method. This means the interface can have multiple default or static methods, but it must have only one unimplemented method that represents the primary functionality of the interface.
  2. Default Method: A functional interface can have multiple default methods, which are methods with implementations provided in the interface itself.
  3. Static Method: A functional interface can also contain multiple static methods, which are methods declared as static within the interface.

Let’s see an example of a functional interface.

@FunctionalInterface
interface MyFunctionalInterface {
    void doSomething(); // Single abstract method

    default void doSomethingElse() {
        System.out.println("Doing something else...");
    }

    default void doAnotherThing() {
        System.out.println("Doing another thing...");
    }

    static void staticMethod() {
        System.out.println("Static method called.");
    }

    static void anotherStaticMethod() {
        System.out.println("Another static method called.");
    }
}

Let’s see examples without using Java 8 Functional interface and with Java 8 functional interface

Example


@FunctionalInterface
interface MyCustomFunctionalInterface {
    void doSomething();
}

// With Java 8
public class WithJava8Example1 {
    public static void main(String[] args) {
        MyCustomFunctionalInterface myFunc = () -> System.out.println("Doing something!");
        myFunc.doSomething();
    }
}

Example of Java 8 Functional interface with lambda expression, Method reference, Stream API.

  • Lambda Expressions:

Use Case 1: Sorting a List of Strings using lambda expressions and Comparator.

import java.util.Arrays;
import java.util.List;

public class LambdaExpressionsExample1 {
    public static void main(String[] args) {
        List<String> names = Arrays.asList("Alice", "Bob", "Charlie", "David");

        // Sorting with lambda expression
        names.sort((s1, s2) -> s1.compareTo(s2));

        System.out.println(names);
    }
}

Output: [Alice, Bob, Charlie, David]

Use Case 2: Calculating the area of a rectangle using lambda expressions and a functional interface.

@FunctionalInterface
interface RectangleAreaCalculator {
    double calculateArea(double length, double width);
}

public class LambdaExpressionsExample2 {
    public static void main(String[] args) {
        RectangleAreaCalculator calculator = (length, width) -> length * width;
        double area = calculator.calculateArea(5.0, 3.0);

        System.out.println("Area of the rectangle: " + area);
    }
}

Output: Area of the rectangle: 15.0

  • Method References:

Use Case 1: Mapping a list of strings to their lengths using method references.

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class MethodReferencesExample1 {
    public static void main(String[] args) {
        List<String> names = Arrays.asList("Alice", "Bob", "Charlie", "David");

        // Using method reference
        List<Integer> lengths = names.stream()
                                     .map(String::length)
                                     .collect(Collectors.toList());

        System.out.println(lengths);
    }
}

Output: [5, 3, 7, 5]

Use Case 2: Converting a list of integers to their corresponding strings using method

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class MethodReferencesExample2 {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);

        // Using method reference
        List<String> numberStrings = numbers.stream()
                                            .map(Object::toString)
                                            .collect(Collectors.toList());

        System.out.println(numberStrings);
    }
}

Output: ["1", "2", "3", "4", "5"]

  • Stream API and Functional Programming: Functional interfaces play a crucial role in the Stream API introduced in Java 8. Operations such as map, filter, reduce, and others in the Stream API take functional interfaces as arguments to define the transformation or condition for elements in a stream.

Use Case 1: Filtering even numbers from a list using the Stream API and Predicate.

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class StreamAPIExample1 {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);

        // Using Stream API and Predicate
        List<Integer> evenNumbers = numbers.stream()
                                           .filter(n -> n % 2 == 0)
                                           .collect(Collectors.toList());

        System.out.println(evenNumbers);
    }
}

Output: [2, 4, 6, 8, 10]

Use Case 2: Summing up a list of integers using the Stream API and reduce.

import java.util.Arrays;
import java.util.List;

public class StreamAPIExample2 {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);

        // Using Stream API and reduce
        int sum = numbers.stream()
                         .reduce(0, Integer::sum);

        System.out.println("Sum: " + sum);
    }
}

Output: Sum: 15

Predefined Java 8 functional interfaces examples

1. Predicate<T>

import java.util.function.Predicate;

public class PredicateExample {
    public static void main(String[] args) {
        Predicate<Integer> isEven = num -> num % 2 == 0;
        
        System.out.println(isEven.test(4)); // Output: true
        System.out.println(isEven.test(7)); // Output: false
    }
}

The Predicate functional interface takes an argument and returns a boolean value. In this example, isEven is a predicate that checks whether a given integer is even.

2. Function<T, R>

import java.util.function.Function;

public class FunctionExample {
    public static void main(String[] args) {
        Function<Integer, String> intToString = num -> "Number: " + num;
        
        String result = intToString.apply(42);
        System.out.println(result); // Output: Number: 42
    }
}

The Function functional interface takes an argument of type T and returns a result of type R. In this example, intToString converts an integer to a string with a prefix.

3. Consumer<T>

import java.util.function.Consumer;

public class ConsumerExample {
    public static void main(String[] args) {
        Consumer<String> printMessage = message -> System.out.println(message);
        
        printMessage.accept("Hello, world!"); // Output: Hello, world!
    }
}

The Consumer functional interface takes an argument of type T and performs some action without returning a result. In this example, printMessage prints the provided message to the console.

4. Supplier<T>

import java.util.function.Supplier;

public class SupplierExample {
    public static void main(String[] args) {
        Supplier<Double> randomValue = () -> Math.random();
        
        System.out.println(randomValue.get()); // Output: (random double value)
    }
}

The Supplier functional interface takes no arguments and returns a result of type T. In this example, randomValue supplies a random double value using the Math.random() method.

5. UnaryOperator<T>

import java.util.function.UnaryOperator;

public class UnaryOperatorExample {
    public static void main(String[] args) {
        UnaryOperator<Integer> square = num -> num * num;
        
        System.out.println(square.apply(5)); // Output: 25
    }
}

The UnaryOperator functional interface takes an argument of type T and returns a result of the same type T. In this example, square calculates the square of an integer.

6. BinaryOperator<T>

import java.util.function.BinaryOperator;

public class BinaryOperatorExample {
    public static void main(String[] args) {
        BinaryOperator<Integer> sum = (a, b) -> a + b;
        
        System.out.println(sum.apply(3, 5)); // Output: 8
    }
}

The BinaryOperator functional interface takes two arguments of type T and returns a result of the same type T. In this example, sum calculates the sum of two integers.

These examples demonstrate how to use various predefined Java 8 functional interfaces to achieve different types of operations using lambda expressions.

Questions and answers related to Java 8 functional interface.

1: What is a functional interface in Java 8? Answer: A functional interface in Java 8 has a single abstract method and serves as the basis for lambda expressions.

2: How is a lambda expression related to a functional interface? Answer: Lambda expressions offer a concise way to implement the single method of a functional interface.

3: What is the purpose of the java.util.function package in Java 8? Answer: The java.util.function package provides built-in functional interfaces that aid functional programming tasks.

4: What are method references, and how do they relate to functional interfaces? Answer: Method references are shortcuts to refer to methods, often used with functional interfaces as compact alternatives to lambda expressions.

5: What are the key differences between Function and Consumer functional interfaces? Answer: Function takes an input and returns a value, while Consumer takes an input and performs an action without returning anything.

6: How do you define a custom functional interface? Answer: You define a custom functional interface by declaring a single abstract method within it.

7: What is the purpose of the @FunctionalInterface annotation? Answer: The @FunctionalInterface annotation ensures that an interface has only one abstract method, highlighting its functional nature.

8: What does the term “SAM interface” mean in the context of functional interfaces? Answer: “SAM” stands for Single Abstract Method. A SAM interface is a functional interface with only one abstract method.

9: What is the advantage of using method references over lambda expressions? Answer: Method references can lead to more concise and readable code when you’re directly referring to an existing method.

10: How do functional interfaces enhance the Java programming paradigm? Answer: Functional interfaces promote a more modular and expressive code style, enabling the adoption of functional programming concepts in Java.

Check Java 8 functional interface docs.

Other Java 8 examples.

That’s all about Java 8 Functional Interface Examples.