
How to improve the performance of primitive types in Java’s Stream Api?
Natan Ferreira
- 0
- 170
Java’s Stream API facilitates code maintenance, is used with lambda expressions and functional paradigm. It offers many features such as filtering a list, among others.
Let’s see an example. For this, I created a class called Product.
public class Product {
private String name;
private int quantity;
public Product(String name, int quantity) {
this.name = name;
this.quantity = quantity;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getQuantity() {
return quantity;
}
public void setQuantity(int quantity) {
this.quantity = quantity;
}
}
I populated a list of products and created the following stream. My goal is to map the quantity, which is a primitive type “int”.
Stream<Integer> quantities = products.stream()
.map(Product::getQuantity);
This returns a Stream because the quantity returns an integer value, which is a primitive type “int“. However, it is not possible to return a Stream of “int“, so the transformation from “int” to “Integer” (Boxing) occurs.
A Stream with a few elements might have an irrelevant performance loss, but a Stream with many elements can result in significant performance loss.
That’s why there’s a special type to use for primitive values of type “int”, which is “IntStream”. To achieve this, we need to use “mapToInt” to return an “IntStream” since “map” returns a “Stream”.
IntStream stream = products.stream()
.mapToInt(Product::getQuantity);

I created an empty list of products and added many products to it.
List<Product> products = new ArrayList<>();
for (int i = 0; i < 10_000_000_0L; i++) {
products.add(new Product("Product "+ i, i));
}
Then I tried to sum using the “map”.
long startTime = System.currentTimeMillis();
Integer sum = products.stream()
.map(Product::getQuantity).reduce(0, Integer::sum);
System.out.println("sum Integer: " + sum);
long endTime = System.currentTimeMillis();
long duration = endTime - startTime;
System.out.println("Execution time sum Integer: " + duration + " milliseconds");
I had the following result.

Then I tried to add using “mapToInt”.
long startTime = System.currentTimeMillis();
int sum = products.stream()
.mapToInt(Product::getQuantity).sum();
System.out.println("sumInt: " + sum);
long endTime = System.currentTimeMillis();
long duration = endTime - startTime;
System.out.println("Execution time sum int: " + duration + " milliseconds");


I was able to notice a difference in execution time using the same list of products, the same computer for both, and the same operating system.
It’s interesting to note that there are also other methods available for other primitive types, such as “mapToLong” and “mapToDouble”.


Author
-
I am a seasoned Full Stack Software Developer with 8+ years of experience, including 6+ years specializing in Java with Spring and Quarkus. My core expertise lies in developing robust RESTful APIs integrated with Cosmos Db, MySQL, and cloud platforms like Azure and AWS. I have extensive experience designing and implementing microservices architectures, ensuring performance and reliability for high-traffic systems. In addition to backend development, I have experience with Angular to build user-friendly interfaces, leveraging my postgraduate degree in frontend web development to deliver seamless and responsive user experiences. My dedication to clean and secure code led me to present best practices to my company and clients, using tools like Sonar to ensure code quality and security. I am a critical thinker, problem solver, and team player, thriving in collaborative environments while tackling complex challenges. Beyond development, I share knowledge through my blog, NatanCode, where I write about Java, Spring, Quarkus, databases, and frontend development. My passion for learning and delivering innovative solutions drives me to excel in every project I undertake.