How do I use the LongUnaryOperator functional interface in Java?

Key Features of LongUnaryOperator

  • It performs operations on long values.
  • It has a single abstract method: long applyAsLong(long operand), which takes one long argument and returns a long result.
  • Additional default and static methods such as andThen() and compose() allow chaining of operations.

Key Methods in LongUnaryOperator

  1. long applyAsLong(long operand)
    • Performs the operation and returns the result.
  2. andThen(LongUnaryOperator after)
    • Returns a composed LongUnaryOperator that applies the current operation, then applies the specified LongUnaryOperator operation.
  3. compose(LongUnaryOperator before)
    • Returns a composed LongUnaryOperator that applies the given operation first, and then the current operation.
  4. static LongUnaryOperator identity()
    • Returns a LongUnaryOperator that returns the input value as-is (acts as an identity function).

Using LongUnaryOperator in Practice

1. Basic Example

Here is a simple example where we use LongUnaryOperator to create a lambda that doubles a number:

package org.kodejava.util.function;

import java.util.function.LongUnaryOperator;

public class LongUnaryOperatorExample {
   public static void main(String[] args) {
      // Lambda to double the value
      LongUnaryOperator doubleValue = operand -> operand * 2;

      long input = 5L;
      // Apply the operation
      long result = doubleValue.applyAsLong(input);
      System.out.println("Result: " + result);
      // Output: 10
   }
}

2. Using Method References

We can use method references to refer to a static or instance method that matches the functional signature:

package org.kodejava.util.function;

import java.util.function.LongUnaryOperator;

public class LongUnaryOperatorMethodRef {
   public static long square(long x) {
      return x * x;
   }

   public static void main(String[] args) {
      // Method reference
      LongUnaryOperator squareOperator = LongUnaryOperatorMethodRef::square;

      long input = 6L;
      // Apply the operation
      long result = squareOperator.applyAsLong(input);
      System.out.println("Result: " + result);
      // Output: 36
   }
}

3. Chaining Operations

The andThen() and compose() methods can be used to chain operations sequentially.

  • andThen(): Executes the current operation first, then the next operation.
  • compose(): Executes the given operation first, then the current operation.

Example:

package org.kodejava.util.function;

import java.util.function.LongUnaryOperator;

public class LongUnaryOperatorChaining {
   public static void main(String[] args) {
      LongUnaryOperator increment = operand -> operand + 1;
      LongUnaryOperator square = operand -> operand * operand;

      // Chain operations: (increment -> square)
      LongUnaryOperator incrementThenSquare = increment.andThen(square);

      long input = 3L;
      // (3 + 1)^2 = 16
      long result1 = incrementThenSquare.applyAsLong(input);
      System.out.println("Increment then Square Result: " + result1);

      // Chain operations: (increment -> square)
      LongUnaryOperator incrementComposeSquare = square.compose(increment);

      // (3 + 1)^2 = 16
      long result2 = incrementComposeSquare.applyAsLong(input);
      System.out.println("Increment compose Increment Result: " + result2);
   }
}

4. Using with Streams

The LongUnaryOperator can also be used in contexts like streams, especially LongStream:

package org.kodejava.util.function;

import java.util.stream.LongStream;
import java.util.function.LongUnaryOperator;

public class LongUnaryOperatorWithStreams {
   public static void main(String[] args) {
      LongUnaryOperator multiplyByThree = operand -> operand * 3;

      LongStream numbers = LongStream.of(1L, 2L, 3L, 4L);
      numbers.map(multiplyByThree).forEach(System.out::println);
      // Output:
      // 3
      // 6
      // 9
      // 12
   }
}

5. Example of identity():

package org.kodejava.util.function;

import java.util.function.LongUnaryOperator;

public class LongUnaryOperatorIdentity {
   public static void main(String[] args) {
      LongUnaryOperator identity = LongUnaryOperator.identity();

      long input = 25L;
      // Input and output are the same
      long result = identity.applyAsLong(input);
      System.out.println("Identity Result: " + result);
      // Output: 25
   }
}

Summary

The LongUnaryOperator is a useful functional interface for performing operations on long values. It is commonly applied for transformations in data processing workflows, functional programming, and stream pipelines. Its ability to compose and chain operations allows for clean and concise code.

How do I use the LongToIntFunction functional interface in Java?

The LongToIntFunction is a functional interface in Java located in the java.util.function package. It represents a function that accepts a long-valued argument and produces an int-valued result. It is a specialized form of functional interface primarily used to avoid boxing when working with primitive types.

Here is a guide on how to use the LongToIntFunction interface:

1. Functional Interface Definition

The LongToIntFunction is defined as:

@FunctionalInterface
public interface LongToIntFunction {
    int applyAsInt(long value);
}

This means:
– It has a single abstract method named applyAsInt(long value), which takes a long as input and returns an int.
– It can be implemented using a lambda expression, method reference, or an anonymous class.


2. Usage Examples

a. Using Lambda Expression:

package org.kodejava.util.function;

import java.util.function.LongToIntFunction;

public class LongToIntFunctionExample {
    public static void main(String[] args) {
        // Example: Converting long to int (e.g., modulo operation)
        // Converts and limits to two-digit values
        LongToIntFunction longToInt = value -> (int) (value % 100);

        long exampleValue = 12345678L;
        int result = longToInt.applyAsInt(exampleValue);

        System.out.println("Result: " + result);
        // Output: 78
    }
}

In this example, the lambda expression converts a long value into an int by taking its modulo.


b. Using Method Reference:

If there’s an existing method compatible with the signature of applyAsInt(long value), we can use a method reference.

package org.kodejava.util.function;

import java.util.function.LongToIntFunction;

public class LongToIntFunctionExample2 {
    public static void main(String[] args) {
        // Example: Using a method reference for a custom conversion
        LongToIntFunction longToInt = LongToIntFunctionExample2::convertLongToInt;

        long exampleValue = 987654321L;
        int result = longToInt.applyAsInt(exampleValue);

        // Custom behavior
        System.out.println("Result: " + result);
    }

    // Custom method matching the LongToIntFunction signature
    public static int convertLongToInt(long value) {
        // Keep the last 3 digits as int
        return (int) (value % 1000);
    }
}

c. Using an Anonymous Class:

We can also use an anonymous class to implement the interface.

package org.kodejava.util.function;

import java.util.function.LongToIntFunction;

public class LongToIntFunctionExample3 {
    public static void main(String[] args) {
        // Anonymous class implementation
        LongToIntFunction longToInt = new LongToIntFunction() {
            @Override
            public int applyAsInt(long value) {
                return (int) (value / 1000);
            }
        };

        long exampleValue = 987654321L;
        int result = longToInt.applyAsInt(exampleValue);

        System.out.println("Result: " + result);
        // Output: 987654
    }
}

3. When to Use

The LongToIntFunction is useful when:
– We work with primitive types (long and int) and want to avoid the unnecessary cost of boxing and unboxing.
– We need to process a long and get an int result in a concise, functional programming style.


4. Real-life Scenario

Imagine we are working with a large dataset containing timestamps represented as long values. We may want to extract part of the timestamp (like the hour or day) as an int:

package org.kodejava.util.function;

import java.time.Instant;
import java.util.function.LongToIntFunction;

public class LongToIntFunctionExample4 {
    public static void main(String[] args) {
        // Extract the hour part from a UNIX timestamp
        LongToIntFunction extractHour =
                timestamp -> Instant.ofEpochSecond(timestamp)
                        .atZone(java.time.ZoneId.systemDefault()).getHour();

        // Current timestamp in seconds
        long currentTimestamp = Instant.now().getEpochSecond();
        int hour = extractHour.applyAsInt(currentTimestamp);

        System.out.println("Current Hour: " + hour);
    }
}

This shows how we can use LongToIntFunction in a practical application.


Summary

  • The LongToIntFunction interface makes it easy to handle the transformation from long to int type.
  • It avoids overhead caused by boxing and unboxing of primitive types.
  • It is often used with lambda expressions, method references, or anonymous classes.

How do I use the LongToDoubleFunction functional interface in Java?

The LongToDoubleFunction is a functional interface in Java that belongs to the java.util.function package. It represents a function that accepts a long value as an argument and produces a result of type double. This functional interface is typically used in scenarios where we want to perform an operation that converts a long input into a double output without having to box or unbox objects.

Here’s how we can use the LongToDoubleFunction interface:

1. Structure of the Interface

The LongToDoubleFunction interface has a single abstract method:

@FunctionalInterface
public interface LongToDoubleFunction {
    double applyAsDouble(long value);
}

This method, applyAsDouble, takes a long as input and returns a double.


2. Example Usage

Example 1: Using a Lambda Expression

We can use a lambda expression to implement the applyAsDouble method:

package org.kodejava.util.function;

import java.util.function.LongToDoubleFunction;

public class LongToDoubleFunctionExample {
    public static void main(String[] args) {
        // Example 1: Convert a long to a double (e.g., divide by 2.5)
        LongToDoubleFunction longToDouble = (value) -> value / 2.5;

        long input = 10L;
        double result = longToDouble.applyAsDouble(input);

        System.out.println("Input: " + input);
        System.out.println("Result: " + result);
    }
}

Output:

Input: 10
Result: 4.0

Example 2: Referencing a Method

We can also use a method reference if we have a method that matches the signature of applyAsDouble.

package org.kodejava.util.function;

import java.util.function.LongToDoubleFunction;

public class LongToDoubleFunctionExample2 {
    public static void main(String[] args) {
        LongToDoubleFunction convertToDouble = LongToDoubleFunctionExample2::convert;

        long input = 15L;
        double result = convertToDouble.applyAsDouble(input);

        System.out.println("Input: " + input);
        System.out.println("Result: " + result);
    }

    public static double convert(long value) {
        return value * 1.2; // Example logic
    }
}

Output:

Input: 15
Result: 18.0

3. Practical Use Cases

  • Mathematical Calculations: Converting long values (like IDs or timestamps) into floating-point representations for calculations.
  • Unit Conversion: Transforming a value in one unit (e.g., seconds as a long) to another (e.g., minutes as a double).
  • Stream API: We can use LongToDoubleFunction with streams, such as LongStream, for functional programming.

Example 3: Using with Streams

package org.kodejava.util.function;

import java.util.function.LongToDoubleFunction;
import java.util.stream.LongStream;

public class LongToDoubleFunctionExample3 {
    public static void main(String[] args) {
        LongToDoubleFunction divideByPi = (value) -> value / Math.PI;

        LongStream.of(10L, 20L, 30L)
                .mapToDouble(divideByPi) // Map each long to a double
                .forEach(System.out::println); // Print each result
    }
}

Output:

3.183098861837907
6.366197723675814
9.549296585513721

4. Key Points

  • It is one of the specialized functional interfaces in Java (LongToIntFunction, DoubleToLongFunction, etc.) to avoid boxing overhead.
  • It is a functional interface, so we can use it with lambda expressions, method references, or anonymous classes.
  • Commonly used with streams for functional-style processing of numbers.

How do I use the LongSupplier functional interface in Java?

The LongSupplier is a functional interface in Java that is part of the java.util.function package. It represents a function that supplies a long-valued result. This interface is particularly useful when we need to generate or provide long values without requiring input arguments.

Since LongSupplier is a functional interface, it has a single abstract method:

  • long getAsLong(): This method is used to return a long value.

Here is how we can use the LongSupplier interface in Java:


Example 1: Basic Usage with Lambda Expression

package org.kodejava.util.function;

import java.util.function.LongSupplier;

public class LongSupplierExample {
    public static void main(String[] args) {
        // Using a LongSupplier to provide the current system time in milliseconds
        LongSupplier currentTimeSupplier = () -> System.currentTimeMillis();

        // Getting a long value
        long currentTime = currentTimeSupplier.getAsLong();
        System.out.println("Current Time in Milliseconds: " + currentTime);
    }
}

Example 2: Using Method References

We can also use method references instead of a lambda if it matches the expected signature of the LongSupplier.

package org.kodejava.util.function;

import java.util.function.LongSupplier;

public class LongSupplierExample2 {
    public static void main(String[] args) {
        // Using method reference
        LongSupplier nanoTimeSupplier = System::nanoTime;

        // Getting a long value
        long nanoTime = nanoTimeSupplier.getAsLong();
        System.out.println("Current Time in Nanoseconds: " + nanoTime);
    }
}

Example 3: Custom Implementation

We can create our own implementation of LongSupplier.

package org.kodejava.util.function;

import java.util.function.LongSupplier;

public class LongSupplierExample3 {
    public static void main(String[] args) {
        // Custom implementation
        LongSupplier randomLongSupplier = new LongSupplier() {
            @Override
            public long getAsLong() {
                // Generate a random long value
                return (long) (Math.random() * 1000);
            }
        };

        // Getting a long value
        long randomValue = randomLongSupplier.getAsLong();
        System.out.println("Random Long Value: " + randomValue);
    }
}

Example 4: Using with Streams

LongSupplier can also be useful when used with streams for generating a sequence of long values.

package org.kodejava.util.function;

import java.util.function.LongSupplier;
import java.util.stream.LongStream;

public class LongSupplierExample4 {
    public static void main(String[] args) {
        // LongSupplier to generate an infinite series of long values
        LongSupplier supplier = new LongSupplier() {
            private long start = 1;

            @Override
            public long getAsLong() {
                return start++;
            }
        };

        // Using the LongSupplier with LongStream.generate
        LongStream.generate(supplier)
                .limit(10) // Limit to 10 values
                .forEach(System.out::println); // Print each value
    }
}

Key Points:

  1. LongSupplier is a no-argument functional interface that always returns a long value.
  2. It is lightweight and simple to use with lambdas, method references, or custom implementations.
  3. Useful in scenarios where we need to continuously or repeatedly generate long values, such as timestamps, counters, or random numbers.

How do I use the LongPredicate functional interface in Java?

In Java, the LongPredicate is a functional interface introduced in Java 8 as part of the java.util.function package. It represents a predicate (a boolean-valued function) that takes a single long value as its input argument.

The key method in LongPredicate is:

boolean test(long value);

Here’s how we can use the LongPredicate functional interface:

Example Usage of LongPredicate

1. Basic Example with Lambda Expression:

package org.kodejava.util.function;

import java.util.function.LongPredicate;

public class LongPredicateExample {
   public static void main(String[] args) {
      // Define a LongPredicate to check if a number is even
      LongPredicate isEven = value -> value % 2 == 0;

      long number = 10;

      // Use the LongPredicate
      if (isEven.test(number)) {
         System.out.println(number + " is even.");
      } else {
         System.out.println(number + " is odd.");
      }
   }
}

2. Combining LongPredicates Using and, or, and negate:

package org.kodejava.util.function;

import java.util.function.LongPredicate;

public class LongPredicateExample2 {
   public static void main(String[] args) {
      // Define two LongPredicates: even and greater than 5
      LongPredicate isEven = value -> value % 2 == 0;
      LongPredicate isGreaterThanFive = value -> value > 5;

      long number = 10;

      // Combine using 'and'
      LongPredicate isEvenAndGreaterThanFive = isEven.and(isGreaterThanFive);
      System.out.println("Is " + number + " even and greater than 5? " +
                         isEvenAndGreaterThanFive.test(number)); // true

      // Combine using 'or'
      LongPredicate isEvenOrGreaterThanFive = isEven.or(isGreaterThanFive);
      System.out.println("Is " + number + " even or greater than 5? " +
                         isEvenOrGreaterThanFive.test(number)); // true

      // Negate the 'isEven' predicate
      LongPredicate isOdd = isEven.negate();
      System.out.println("Is " + number + " odd? " + isOdd.test(number)); // false
   }
}

3. Using with Streams (e.g., LongStream):

If we use the LongPredicate with streams, it can be helpful for filtering LongStream.

package org.kodejava.util.function;

import java.util.function.LongPredicate;
import java.util.stream.LongStream;

public class LongPredicateWithStreams {
   public static void main(String[] args) {
      // Generate a stream of numbers from 1 to 10
      LongStream stream = LongStream.rangeClosed(1, 10);

      // Define a LongPredicate to filter even numbers
      LongPredicate isEven = value -> value % 2 == 0;

      // Use 'filter' with LongPredicate
      stream.filter(isEven)
              .forEach(System.out::println); // Prints 2, 4, 6, 8, 10
   }
}

4. Method Reference with LongPredicate:

We can also use method references if we have a method that matches the signature of the test method:

package org.kodejava.util.function;

import java.util.function.LongPredicate;

public class LongPredicateMethodReference {
   public static void main(String[] args) {
      LongPredicate isPositive = LongPredicateMethodReference::isPositive;

      long number = 5;
      // true
      System.out.println("Is " + number + " positive? " + isPositive.test(number));
   }

   // Method to check if a number is positive
   public static boolean isPositive(long value) {
      return value > 0;
   }
}

Key Points about LongPredicate

  • The functional method in LongPredicate is boolean test(long value). This method evaluates the predicate against the given long value and returns a boolean result.
  • It is often used in lambda expressions or method references.
  • We can combine multiple LongPredicate instances using and, or, and negate methods.

Summary

  • Use LongPredicate when working with primitive long arguments in functional programming scenarios.
  • It helps to avoid boxing overhead compared to Predicate<Long>.
  • We can define simple or complex conditions, combine predicates using and, or, and negate, and use them with streams for concise and readable code.