Java 8 introduces Stream, which is a new abstract layer, and some new additional packages in Java 8 called java.util.stream. A Stream is a sequence of components that can be processed sequentially. These packages include classes, interfaces, and enum to allow functional-style operations on the elements.
The stream can be used by importing java.util.stream package. Stream API is used to process collections of objects. Streams are designed to be efficient and can support improving your program's performance by allowing you to avoid unnecessary loops and iterations. Streams can be used for filtering, collecting, printing, and converting from one data structure to another, etc.

This Java 8 Stream Tutorial will cover all the basic to advanced concepts of Java 8 stream like Java 8 filter and collect operations, and real-life examples of Java 8 streams.
Prerequisites for Java Stream
Before proceeding to Java 8, it's recommended to have a basic knowledge of Java 8 and its important concepts such as lambda expression, Optional, method references, etc.
Note:
- If we want to represent a group of objects as a single entity then we should go for collection.
- But if we want to process objects from the collection then we should go for streams.
If we want to use the concept of streams then stream() is the method to be used. Stream is available as an interface.
Syntax:
Stream s = c.stream();
In the above pre-tag, 'c' refers to the collection. So on the collection, we are calling the stream() method and at the same time, we are storing it as the Stream object. Henceforth, this way we are getting the Stream object.
Note: Streams are present in java's utility package named java.util.stream
Let us now start with the basic components involved in streams. They as listed as follows:
- Sequence of Elements
- Source
- Aggregate Operations
- Pipelining
- Internal iteration
Features of Java Stream
- A stream is not a data structure instead it takes input from the Collections, Arrays, or I/O channels.
- Streams don’t change the original data structure, they only provide the result as per the pipelined methods.
- Each intermediate operation is lazily executed and returns a stream as a result, hence various intermediate operations can be pipelined. Terminal operations mark the end of the stream and return the result.
Before moving ahead in the concept consider an example in which we are having ArrayList of integers, and we suppose we apply a filter to get only even numbers from the object inserted.

How does Stream Work Internally?
In streams,
- To filter out from the objects we do have a function named filter()
- To impose a condition we do have a logic of predicate which is nothing but a functional interface. Here function interface can be replaced by a random expression. Hence, we can directly impose the condition check-in our predicate.
- To collect elements we will be using Collectors.toList() to collect all the required elements.
- Lastly, we will store these elements in a List and display the outputs on the console.
Example
Java
// Java Program to illustrate FILTER
// & COLLECT Operations
import java.io.*;
import java.util.*;
import java.util.stream.*;
// Main class
public class GFG {
// Main driver method
public static void main(String[] args)
{
// Creating an ArrayList object of integer type
ArrayList<Integer> al = new ArrayList<Integer>();
// Inserting elements to ArrayList class object
// Custom input integer numbers
al.add(2);
al.add(6);
al.add(9);
al.add(4);
al.add(20);
// First lets print the collection
System.out.println("Printing the collection : "
+ al);
// Printing new line for better output readability
System.out.println();
// Stream operations
// 1. Getting the stream from this collection
// 2. Filtering out only even elements
// 3. Collecting the required elements to List
List<Integer> ls
= al.stream()
.filter(i -> i % 2 == 0)
.collect(Collectors.toList());
// Print the collection after stream operation
// as stored in List object
System.out.println(
"Printing the List after stream operation : "
+ ls);
}
}
OutputPrinting the collection : [2, 6, 9, 4, 20]
Printing the List after stream operation : [2, 6, 4, 20]
Explanation of the above program:
In our collection object, we were having elements entered using the add() operation. After processing the object in which they were stored through streams we impose a condition in the predicate of streams to get only even elements, we get elements in the object as per our requirement. Hence, streams helped us this way in processing over-processed collection objects.
Various Core Operations Over Streams
There are broadly 3 types of operations that are carried over streams namely as follows as depicted from the image shown above:
- Intermediate operations
- Terminal operations
- Short-circuit operations

Let us do discuss out intermediate operations here only in streams to a certain depth with the help of an example in order to figure out other operations via theoretical means.
1. Intermediate Operations:
Intermediate operations transform a stream into another stream. Some common intermediate operations include:
- filter(): Filters elements based on a specified condition.
- map(): Transforms each element in a stream to another value.
- sorted(): Sorts the elements of a stream.
All three of them are discussed below as they go hand in hand in nearly most of the scenarios and to provide better understanding by using them later by implementing in our clean Java programs below. As we already have studied in the above example of which we are trying to filter processed objects can be interpreted as filter() operation operated over streams.
2. Terminal Operations
Terminal Operations are the operations that on execution return a final result as an absolute value.
- collect(): It is used to return the result of the intermediate operations performed on the stream.
- forEach(): It iterates all the elements in a stream.
- reduce(): It is used to reduce the elements of a stream to a single value.
3. Short Circuit Operations
Short-circuit operations provide performance benefits by avoiding unnecessary computations when the desired result can be obtained early. They are particularly useful when working with large or infinite streams.
- anyMatch(): it checks the stream if it satisfies the given condition.
- findFirst(): it checks the element that matches a given condition and stops processing when it finds it.
Note: They are lazy, meaning they are not executed until a terminal operation is invoked.
Later on from that processed filtered elements of objects, we are collecting the elements back to List using Collectors for which we have imported a specific package named java.util.stream with the help of Collectors.toList() method. This is referred to as collect() operation in streams so here again we won't be taking an example to discuss them out separately.
Example:
Java
// Java program to illustrate Intermediate Operations
// in Streams
// Importing required classes
import java.io.*;
import java.util.*;
import java.util.stream.*;
// Main class
class Test {
// Main driver method
public static void main(String[] args)
{
// Creating an integer Arraylist to store marks
ArrayList<Integer> marks = new ArrayList<Integer>();
// These are marks of the students
// Considering 5 students so input entries
marks.add(30);
marks.add(78);
marks.add(26);
marks.add(96);
marks.add(79);
// Printing the marks of the students before grace
System.out.println(
"Marks of students before grace : " + marks);
// Now we want to grace marks by 6
// using the streams to process over processing
// collection
// Using stream, we map every object and later
// collect to List
// and store them
List<Integer> updatedMarks
= marks.stream()
.map(i -> i + 6)
.collect(Collectors.toList());
// Printing the marks of the students after grace
System.out.println(
"Marks of students after grace : "
+ updatedMarks);
}
}
OutputMarks of students before grace : [30, 78, 26, 96, 79]
Marks of students after grace : [36, 84, 32, 102, 85]
Note: For every object if there is urgency to do some operations be it square, double or any other than only we need to use map() function operation else try to use filter() function operation.
Lazy Evaluation
Lazy Evaluation is the concept in Java Streams where computation on the source data is only performed when the terminal operation is initiated, and source elements are consumed only as needed. It is called lazy because intermediate operations are not evaluated until a terminal operation is invoked. Some terminal operations may also short-circuit and processing only as many elements as needed to return a result.
Now geeks you are well aware of 'why' streams were introduced, but you should be wondering 'where' to use them. The answer is very simple as we do use them too often in our day-to-day life. Hence, the geek in simpler words we say directly lands p on wherever the concept of the collection is applicable, stream concept can be applied there.
Java Stream: Real-life Examples
Example 1:
In general, daily world, whenever the data is fetched from the database, it is more likely we will be using collection so there stream concept is must apply to deal with processed data.

Now we will be discussing real-time examples to interrelate streams in our life. Here we will be taking the most widely used namely as follows:
- Streams in a Grocery store
- Streams in mobile networking
1. Streams in a Grocery store

The above pictorial image has been provided is implemented in streams which are as follows:
List<Integer> transactionsIds =
transactions.stream()
.filter(t -> t.getType() == Transaction.GROCERY)
.sorted(comparing(Transaction::getValue).reversed())
.map(Transaction::getId)
.collect(toList());
2. Streams in mobile networking
Similarly, we can go for another widely used concept which is our dealing with our mobile numbers. Here we will not be proposing listings, simply will be demonstrating how the stream concept is invoked in mobile networking by various service providers across the globe.
Collection can hold any number of object so let 'mobileNumber' be a collection and let it be holding various mobile numbers say it be holding 100+ numbers as objects. Suppose now the only carrier named 'Airtel' whom with which we are supposed to send a message if there is any migration between states in a country. So here streams concept is applied as if while dealing with all mobile numbers we will look out for this carrier using the filter() method operation of streams. In this way, we are able to deliver the messages without looking out for all mobile numbers and then delivering the message which senses impractical if done so as by now we are already too late to deliver. In this way these intermediate operations namely filter(), collect(), map() help out in the real world. Processing becomes super simpler which is the necessity of today's digital world.
Hope by now you the users come to realize the power of streams in Java as if we have to do the same task we do need to map corresponding to every object, increasing in code length, and decreasing the optimality of our code. With the usage of streams, we are able to in a single line irrespective of elements contained in the object as with the concept of streams we are dealing with the object itself.
Note: filter, sorted, and map, which can be connected together to form a pipeline.
What is a Pipeline?
A Stream Pipeline is a concept of chaining operations together Terminal Operations and Intermediate Operations. A Pipeline contains a stream source, which is further followed by zero or more intermediate operations, and a terminal operation.
Java Stream Operations
Method Types and Pipelines
Methods are of two types in Stream as mentioned below:
- Terminal Operations
- Intermediate Operations
Terminal Operations
These are the operations that after consumed can't further be used. There are few operations mentioned below:
forEach performs an action for each element of the stream. Stream forEach is a terminal operation.
Syntax
void forEach(Consumer<? super T> action)
Stream toArray() returns an array containing the elements of this stream. After the terminal operation is performed, the stream pipeline is considered consumed, and can no longer be used.
Syntax
Object[] toArray()
min and max return the min and max elements from the stream.
Syntax
Optional<T> min(Comparator<? super T> comparator)
Optional<T> max(Comparator<? super T> comparator)
Where, Optional is a container object which may or may not contain a non-null value, and T is the type of object that may be compared by this comparator.
Intermediate Operations
It returns a new stream that can be further processed. There are certain operations mentioned below:
Stream filter returns a stream consisting of the elements of this stream that match the given predicate.
Syntax
Stream<T> filter(Predicate<? super T> predicate)
distinct() returns a stream consisting of distinct elements in a stream. distinct() is the method of Stream interface.
Syntax
Stream<T> distinct()
Where Stream is an interface and the function returns a stream consisting of distinct elements.
Stream sorted() returns a stream consisting of the elements of this stream, sorted according to natural order. For ordered streams, the sort method is stable but for unordered streams, no stability is guaranteed.
Syntax
Stream<T> sorted()
where Stream is an interface and T is the type of stream elements.
Comparison-Based Stream Operations
Comparison Based Stream Operations are the used for comparing, sorting, and ordering elements within a stream. There are certain examples of Comparison Based Stream Operations mentioned below:
- Sorted
- min and max
- distinct
Java Stream Specializations
As there are primitive data types or specializations like int, long and double. Similarly, streams have IntStream, LongStream, and DoubleStream. These are convenient for making performing transactions with numerical primitives.
1. Specialized Operations
Specialized streams provide additional operations as compared to the standard Stream – which are quite convenient when dealing with numbers.
2. Reduction Operations
Reduce Operation applies a binary operator, it takes a sequence of input elements and combines them to a single summary result. It is all done where first argument to the operator is the return value of the previous application and second argument is the current stream element.
Parallel Streams
Parallel Streams are the type of streams that can perform operations concurrently on multiple threads. These Streams are meant to make use of multiple processors or cores available to speed us the processing speed. There are two methods to create parallel streams are mentioned below:
- Using the parallel() method on a stream
- Using parallelStream() on a Collection
To know more about Parallel Streams refer to this link.
Infinite Streams
Infinite Streams are the type of Streams that can produce unbounded(Infnite) number of elements. These Streams are useful when you need to work with the data sources that are not finite.
Java Stream: File Operation
In this section, we see how to utilize Java stream in file I/O operation.
1. File Read Operation
Let's understand file read operation through the given example
Java
// Java Program to demonstrate
// File Read Operation
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;
class GFG {
// Method to filter strings of a given length and
// convert them to uppercase
private static List<String>
filterAndConvertToUpper(Stream<String> stream,
int length)
{
return stream.filter(s -> s.length() == length)
.map(String::toUpperCase)
.collect(Collectors.toList());
}
public static void main(String[] args)
{
// Replace with the
// actual file path
String fileName = "path/to/your/file.txt";
// Step 1: Create a Stream of lines from the
// file
try (Stream<String> lines
= Files.lines(Paths.get(fileName))) {
List<String> filteredStrings
= filterAndConvertToUpper(lines, 5);
System.out.println(
"Filtered strings with length 5 (converted to uppercase): "
+ filteredStrings);
}
catch (IOException e) {
e.printStackTrace();
}
}
}
Input:
Geeks
gfg
geeks
geeksforgeeks
Coder
Guys
Output:
Filtered strings with length 5 (converted to uppercase): [GEEKS, GEEKS, CODER]
2. File Write Operation
Let's understand file write operation through the given example
Java
// Java Program to demonstrate
// File Write Operation
import java.io.*;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.stream.Stream;
// Driver Class
class GFG {
// main function
public static void main(String[] args)
{
String[] words
= { "Geeks", "for", "Geeks", "Hello", "World" };
// Replace with the
// actual file path
String fileName = "path/to/your/file.txt";
// Step 1: Create a PrintWriter to write to the
// file
try (PrintWriter pw
= new PrintWriter(Files.newBufferedWriter(
Paths.get(fileName)))) {
// Step 2: Use Stream to write each word to the
// file
Stream.of(words).forEach(pw::println);
// Step 3: Print success message to the console
System.out.println(
"Words written to the file successfully.");
}
catch (IOException e) {
// Step 4: Handle any IO exception that occurs
// during the file writing process
e.printStackTrace();
}
}
}
Output:
Words written to the file successfully.
Java Streams Improvements in Java 9
As we know, Java 8 introduced a Java stream to the world which provided a powerful way to process collections of data. However, the following version of the language also contributed to the feature. Thereafter, some improvements and features were added to the Java stream in JAVA 9. In this section, we will provide an overview of the advancements introduced by Java 9 to the Streams API like takeWhile, dropWhile, iterate, ofNullable, etc.
Conclusion
Java Streams offer a powerful way to handle data in Java. They allow developers to write more readable and concise code for processing collections. By using Java Streams, you can easily filter, map, and reduce data with simple and expressive methods. This not only makes your code easier to maintain but also helps improve performance by taking advantage of parallel processing. If you're looking to make your data operations more efficient and straightforward, learning Java Streams is definitely worth the effort.
Similar Reads
Java Tutorial Java is a high-level, object-oriented programming language used to build web apps, mobile applications, and enterprise software systems. It is known for its Write Once, Run Anywhere capability, which means code written in Java can run on any device that supports the Java Virtual Machine (JVM).Java s
10 min read
Basics
Introduction to JavaJava is a high-level, object-oriented programming language developed by Sun Microsystems in 1995. It is platform-independent, which means we can write code once and run it anywhere using the Java Virtual Machine (JVM). Java is mostly used for building desktop applications, web applications, Android
4 min read
Java Programming BasicsJava is one of the most popular and widely used programming language and platform. A platform is an environment that helps to develop and run programs written in any programming language. Java is fast, reliable and secure. From desktop to web applications, scientific supercomputers to gaming console
4 min read
Java MethodsJava Methods are blocks of code that perform a specific task. A method allows us to reuse code, improving both efficiency and organization. All methods in Java must belong to a class. Methods are similar to functions and expose the behavior of objects.Example: Java program to demonstrate how to crea
8 min read
Access Modifiers in JavaIn Java, access modifiers are essential tools that define how the members of a class, like variables, methods, and even the class itself can be accessed from other parts of our program. They are an important part of building secure and modular code when designing large applications. Understanding de
7 min read
Arrays in JavaArrays in Java are one of the most fundamental data structures that allow us to store multiple values of the same type in a single variable. They are useful for storing and managing collections of data. Arrays in Java are objects, which makes them work differently from arrays in C/C++ in terms of me
15+ min read
Java StringsIn Java, a String is the type of object that can store a sequence of characters enclosed by double quotes, and every character is stored in 16 bits, i.e., using UTF 16-bit encoding. A string acts the same as an array of characters. Java provides a robust and flexible API for handling strings, allowi
9 min read
Regular Expressions in JavaIn Java, Regular Expressions or Regex (in short) in Java is an API for defining String patterns that can be used for searching, manipulating, and editing a string in Java. Email validation and passwords are a few areas of strings where Regex is widely used to define the constraints. Regular Expressi
7 min read
OOPs & Interfaces
Classes and Objects in JavaIn Java, classes and objects are basic concepts of Object Oriented Programming (OOPs) that are used to represent real-world concepts and entities. The class represents a group of objects having similar properties and behavior, or in other words, we can say that a class is a blueprint for objects, wh
11 min read
Java ConstructorsIn Java, constructors play an important role in object creation. A constructor is a special block of code that is called when an object is created. Its main job is to initialize the object, to set up its internal state, or to assign default values to its attributes. This process happens automaticall
10 min read
Java OOP(Object Oriented Programming) ConceptsJava Object-Oriented Programming (OOPs) is a fundamental concept in Java that every developer must understand. It allows developers to structure code using classes and objects, making it more modular, reusable, and scalable.The core idea of OOPs is to bind data and the functions that operate on it,
13 min read
Java PackagesPackages in Java are a mechanism that encapsulates a group of classes, sub-packages, and interfaces. Packages are used for: Prevent naming conflicts by allowing classes with the same name to exist in different packages, like college.staff.cse.Employee and college.staff.ee.Employee.They make it easie
8 min read
Java InterfaceAn Interface in Java programming language is defined as an abstract type used to specify the behaviour of a class. An interface in Java is a blueprint of a behaviour. A Java interface contains static constants and abstract methods. Key Properties of Interface:The interface in Java is a mechanism to
12 min read
Collections
Exception Handling
Java Exception HandlingException handling in Java allows developers to manage runtime errors effectively by using mechanisms like try-catch block, finally block, throwing Exceptions, Custom Exception handling, etc. An Exception is an unwanted or unexpected event that occurs during the execution of a program, i.e., at runt
10 min read
Java Try Catch BlockA try-catch block in Java is a mechanism to handle exceptions. This make sure that the application continues to run even if an error occurs. The code inside the try block is executed, and if any exception occurs, it is then caught by the catch block.Example: Here, we are going to handle the Arithmet
4 min read
Java final, finally and finalizeIn Java, the final, finally, and finalize keywords play an important role in exception handling. The main difference between final, finally, and finalize is listed below:final: The final is the keyword that can be used for immutability and restrictions in variables, methods, and classes.finally: The
4 min read
Chained Exceptions in JavaChained Exceptions in Java allow associating one exception with another, i.e. one exception describes the cause of another exception. For example, consider a situation in which a method throws an ArithmeticException because of an attempt to divide by zero.But the root cause of the error was an I/O f
3 min read
Null Pointer Exception in JavaA NullPointerException in Java is a RuntimeException. It occurs when a program attempts to use an object reference that has the null value. In Java, "null" is a special value that can be assigned to object references to indicate the absence of a value.Reasons for Null Pointer ExceptionA NullPointerE
5 min read
Exception Handling with Method Overriding in JavaIn Java, an exception is an unwanted or unexpected event that occurs during a program's execution, i.e., at runtime, and disrupts the normal flow of the programâs instructions. Exception handling in Java handles runtime errors and helps maintain the program's normal flow by using constructs like try
5 min read
Java Advanced
Java Multithreading TutorialThreads are the backbone of multithreading. We are living in the real world which in itself is caught on the web surrounded by lots of applications. With the advancement in technologies, we cannot achieve the speed required to run them simultaneously unless we introduce the concept of multi-tasking
15+ min read
Synchronization in JavaIn multithreading, synchronization is important to make sure multiple threads safely work on shared resources. Without synchronization, data can become inconsistent or corrupted if multiple threads access and modify shared variables at the same time. In Java, it is a mechanism that ensures that only
10 min read
File Handling in JavaIn Java, with the help of File Class, we can work with files. This File Class is inside the java.io package. The File class can be used to create an object of the class and then specifying the name of the file.Why File Handling is Required?File Handling is an integral part of any programming languag
6 min read
Java Method ReferencesIn Java, a method is a collection of statements that perform some specific task and return the result to the caller. A method reference is the shorthand syntax for a lambda expression that contains just one method call. In general, one does not have to pass arguments to method references.Why Use Met
9 min read
Java 8 Stream TutorialJava 8 introduces Stream, which is a new abstract layer, and some new additional packages in Java 8 called java.util.stream. A Stream is a sequence of components that can be processed sequentially. These packages include classes, interfaces, and enum to allow functional-style operations on the eleme
15+ min read
Java NetworkingWhen computing devices such as laptops, desktops, servers, smartphones, and tablets and an eternally-expanding arrangement of IoT gadgets such as cameras, door locks, doorbells, refrigerators, audio/visual systems, thermostats, and various sensors are sharing information and data with each other is
15+ min read
JDBC TutorialJDBC stands for Java Database Connectivity. JDBC is a Java API or tool used in Java applications to interact with the database. It is a specification from Sun Microsystems that provides APIs for Java applications to communicate with different databases. Interfaces and Classes for JDBC API comes unde
12 min read
Java Memory ManagementJava memory management is a fundamental concept that involves the automatic allocation and deallocation of objects, managed by the Java Virtual Machine (JVM). The JVM uses a garbage collector to automatically remove unused objects, freeing up memory in the background. This eliminates the need for de
5 min read
Garbage Collection in JavaGarbage collection in Java is an automatic memory management process that helps Java programs run efficiently. Java programs compile to bytecode that can be run on a Java Virtual Machine (JVM). When Java programs run on the JVM, objects in the heap are created, which is a portion of memory dedicated
7 min read
Memory Leaks in JavaIn programming, a memory leak happens when a program keeps using memory but does not give it back when it's done. It simply means the program slowly uses more and more memory, which can make things slow and even stop working. Working of Memory Management in JavaJava has automatic garbage collection,
3 min read
Practice Java
Java Interview Questions and AnswersJava is one of the most popular programming languages in the world, known for its versatility, portability, and wide range of applications. Java is the most used language in top companies such as Uber, Airbnb, Google, Netflix, Instagram, Spotify, Amazon, and many more because of its features and per
15+ min read
Java Programs - Java Programming ExamplesIn this article, we will learn and prepare for Interviews using Java Programming Examples. From basic Java programs like the Fibonacci series, Prime numbers, Factorial numbers, and Palindrome numbers to advanced Java programs.Java is one of the most popular programming languages today because of its
8 min read
Java Exercises - Basic to Advanced Java Practice Programs with SolutionsLooking for Java exercises to test your Java skills, then explore our topic-wise Java practice exercises? Here you will get 25 plus practice problems that help to upscale your Java skills. As we know Java is one of the most popular languages because of its robust and secure nature. But, programmers
7 min read
Java Quiz | Level Up Your Java SkillsThe best way to scale up your coding skills is by practicing the exercise. And if you are a Java programmer looking to test your Java skills and knowledge? Then, this Java quiz is designed to challenge your understanding of Java programming concepts and assess your excellence in the language. In thi
1 min read
Top 50 Java Project Ideas For Beginners and Advanced [Update 2025]Java is one of the most popular and versatile programming languages, known for its reliability, security, and platform independence. Developed by James Gosling in 1982, Java is widely used across industries like big data, mobile development, finance, and e-commerce.Building Java projects is an excel
15+ min read