목차
1. 스트림 API란 무엇인가?
2. 스트림의 장점
3. 스트림의 기본 구조와 특징
- 스트림의 생성
- 중간 연산과 최종 연산
- 스트림의 특성
4. 스트림 API의 사용 방법
- 필터링 (Filtering)
- 매핑 (Mapping)
- 정렬 (Sorting)
- 집계 (Reduction)
- 수집 (Collecting)
5. 스트림의 병렬 처리
6. 스트림의 한계와 주의사항
7. 예제와 분석
8. 결론 및 추가 학습 자료
1. 스트림 API란 무엇인가?
스트림 API는 자바 8에서 도입된 기능으로, 데이터 소스를 추상화하여 일련의 연산을 수행할 수 있는 강력한 도구입니다. 스트림 API는 컬렉션, 배열 등의 데이터 소스를 효율적으로 처리하기 위해 사용되며, 선언형 프로그래밍 스타일로 데이터를 다룰 수 있습니다. 스트림은 데이터의 흐름을 나타내며, 이 흐름을 통해 데이터를 필터링, 변환, 집계, 정렬하는 등의 작업을 수행할 수 있습니다.
스트림 API는 함수형 프로그래밍의 개념을 자바에 도입함으로써, 반복적인 데이터 처리 작업을 간결하고 명확하게 표현할 수 있도록 해줍니다.
2. 스트림의 장점
스트림 API를 사용하면 다음과 같은 주요 장점을 얻을 수 있습니다:
- 간결성: 스트림 API를 사용하면 복잡한 반복문을 제거하고, 선언형으로 데이터를 처리할 수 있어 코드가 간결해집니다.
- 가독성 향상: 스트림을 사용하면 코드의 의도를 명확하게 나타낼 수 있어 가독성이 향상됩니다.
- 병렬 처리 용이성: 스트림은 병렬 처리를 간단하게 구현할 수 있는 메서드를 제공하여, 데이터를 빠르게 처리할 수 있습니다.
- 지연 연산: 스트림은 지연 연산을 통해 필요할 때만 데이터를 처리하여, 불필요한 연산을 줄이고 성능을 최적화합니다.
3. 스트림의 기본 구조와 특징
스트림 API의 기본 구조는 데이터 소스로부터 스트림을 생성하고, 일련의 중간 연산을 통해 데이터를 변환한 후, 최종 연산을 통해 결과를 얻는 방식입니다. 스트림은 데이터 소스의 요소들을 순차적으로 처리하는 일종의 데이터 파이프라인이라고 할 수 있습니다.
스트림의 생성
스트림은 다양한 방법으로 생성할 수 있습니다. 가장 일반적인 방법은 컬렉션이나 배열에서 스트림을 생성하는 것입니다. 'Stream.of()' 메서드나 'Arrays.stream()' 메서드를 사용하여 배열에서 스트림을 생성할 수 있으며, 'Collection' 인터페이스의 'stream()' 메서드를 사용하여 컬렉션에서 스트림을 생성할 수 있습니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamCreationExample {
public static void main(String[] args) {
// 리스트에서 스트림 생성
List<String> names = Arrays.asList("Alice", "Bob", "Charlie");
Stream<String> nameStream = names.stream();
// 배열에서 스트림 생성
String[] nameArray = {"David", "Eve", "Frank"};
Stream<String> arrayStream = Arrays.stream(nameArray);
// Stream.of()를 사용한 스트림 생성
Stream<String> streamOfNames = Stream.of("George", "Helen", "Ian");
nameStream.forEach(System.out::println);
arrayStream.forEach(System.out::println);
streamOfNames.forEach(System.out::println);
}
}
설명:
- 'names.stream()'은 리스트에서 스트림을 생성합니다.
- 'Arrays.stream(nameArray)'는 배열에서 스트림을 생성합니다.
- 'Stream.of()'는 개별 요소로부터 스트림을 생성합니다.
중간 연산과 최종 연산
스트림 연산은 중간 연산(Intermediate Operations)과 최종 연산(Terminal Operations)으로 구분됩니다.
- 중간 연산: 스트림을 반환하며, 연속적으로 연결할 수 있습니다. 예로는 'filter()', 'map()', 'sorted()' 등이 있습니다.
- 최종 연산: 스트림을 소비하며, 연산이 완료되면 더 이상 스트림을 사용할 수 없습니다. 예로는 'forEach()', 'reduce()', 'collect()' 등이 있습니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamOperationsExample {
public static void main(String[] args) {
List<String> names = Arrays.asList("Alice", "Bob", "Charlie", "David", "Eve");
// 중간 연산: 필터링 및 매핑
Stream<String> filteredStream = names.stream()
.filter(name -> name.startsWith("A"))
.map(String::toUpperCase);
// 최종 연산: 출력
filteredStream.forEach(System.out::println);
}
}
설명:
- 'filter()'는 이름이 "A"로 시작하는 요소만 필터링하는 중간 연산입니다.
- 'map()'은 필터링된 요소를 대문자로 변환하는 중간 연산입니다.
- 'forEach()'는 최종 연산으로, 스트림의 각 요소를 출력합니다.
스트림의 특성
스트림은 다음과 같은 특성을 가집니다:
- 데이터 소스를 변경하지 않음: 스트림 연산은 데이터 소스를 변경하지 않고, 새로운 스트림을 생성하여 데이터를 처리합니다.
- 내부 반복: 스트림은 내부적으로 데이터를 반복 처리하며, 개발자가 직접 반복문을 작성할 필요가 없습니다.
- 지연 연산: 스트림의 중간 연산은 실제로 최종 연산이 호출될 때까지 실행되지 않습니다. 이를 통해 불필요한 연산을 줄일 수 있습니다.
4. 스트림 API의 사용 방법
스트림 API는 다양한 연산을 통해 데이터를 처리할 수 있습니다. 다음은 스트림 API의 주요 사용 방법입니다:
필터링 (Filtering)
필터링은 'filter()' 메서드를 사용하여 조건에 맞는 요소만을 추출하는 작업입니다. 이 작업은 중간 연산으로, 스트림을 반환합니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamFilteringExample {
public static void main(String[] args) {
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
// 짝수만 필터링
List<Integer> evenNumbers = numbers.stream()
.filter(n -> n % 2 == 0)
.collect(Collectors.toList());
System.out.println("Even numbers: " + evenNumbers);
}
}
설명:
- 'filter()' 메서드는 짝수만을 추출하여 새로운 리스트로 수집합니다.
매핑 (Mapping)
매핑은 'map()' 메서드를 사용하여 스트림의 각 요소를 다른 형태로 변환하는 작업입니다. 이 작업도 중간 연산으로, 스트림을 반환합니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamMappingExample {
public static void main(String[] args) {
List<String> names = Arrays.asList("Alice", "Bob", "Charlie");
// 각 이름을 대문자로 변환
List<String> upperCaseNames = names.stream()
.map(String::toUpperCase)
.collect(Collectors.toList());
System.out.println("Uppercase names: " + upperCaseNames);
}
}
설명:
- 'map()' 메서드는 스트림의 각 요소를 대문자로 변환하여 새로운 리스트로 수집합니다.
정렬 (Sorting)
정렬은 'sorted()' 메서드를 사용하여 스트림의 요소를 정렬하는 작업입니다. 기본적으로 오름차순으로 정렬되지만, Comparator를 사용하여 사용자 정의 정렬을 할 수도 있습니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamSortingExample {
public static void main(String[] args) {
List<String> names = Arrays.asList("Charlie", "Alice", "Bob");
// 이름을 알파벳 순으로 정렬
List<String> sortedNames = names.stream()
.sorted()
.collect(Collectors.toList());
System.out.println("Sorted names: " + sortedNames);
}
}
설명:
- 'sorted()' 메서드는 스트림의 요소를 기본적으로 오름차순으로 정렬하여 새로운 리스트로 수집합니다.
집계 (Reduction)
집계는 스트림의 모든 요소를 하나의 결과로 축약하는 작업입니다. 'reduce()' 메서드를 사용하여 이를 수행할 수 있습니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamReductionExample {
public static void main(String[] args) {
List<Integer> numbers = Arrays.asList(
1, 2, 3, 4, 5);
// 모든 숫자의 합 계산
int sum = numbers.stream()
.reduce(0, Integer::sum);
System.out.println("Sum: " + sum);
}
}
설명:
- 'reduce()' 메서드는 초기값 '0'과 함께 스트림의 모든 요소를 합산하여 결과를 반환합니다.
수집 (Collecting)
수집은 스트림의 요소를 컬렉션으로 모으는 작업입니다. 'collect()' 메서드를 사용하여 리스트, 세트, 맵 등으로 결과를 수집할 수 있습니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class StreamCollectingExample {
public static void main(String[] args) {
List<String> names = Arrays.asList("Alice", "Bob", "Charlie", "David");
// 이름을 콤마로 구분된 문자열로 연결
String result = names.stream()
.collect(Collectors.joining(", "));
System.out.println("Joined names: " + result);
}
}
설명:
- 'collect()' 메서드는 스트림의 요소를 콤마로 구분된 문자열로 연결합니다.
5. 스트림의 병렬 처리
스트림은 병렬 처리를 쉽게 구현할 수 있는 메서드를 제공합니다. 'parallelStream()' 메서드를 사용하면 스트림의 요소를 병렬로 처리할 수 있습니다. 병렬 스트림은 멀티코어 시스템에서 성능을 향상시킬 수 있지만, 데이터 간의 의존성이 있을 경우 주의해서 사용해야 합니다.
예제 코드:
import java.util.*;
import java.util.stream.*;
public class ParallelStreamExample {
public static void main(String[] args) {
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
// 병렬 스트림으로 처리
int sum = numbers.parallelStream()
.reduce(0, Integer::sum);
System.out.println("Sum (parallel): " + sum);
}
}
설명:
- 'parallelStream()' 메서드를 사용하여 병렬로 스트림을 처리하여 합산 결과를 출력합니다.
6. 스트림의 한계와 주의사항
스트림은 강력한 도구지만, 사용 시 몇 가지 주의사항이 있습니다:
- 한 번만 소비 가능: 스트림은 일회용입니다. 한 번 사용된 스트림은 다시 사용할 수 없으며, 새로운 스트림을 생성해야 합니다.
- 외부 상태 변경 지양: 스트림 내부에서 외부 상태를 변경하는 작업은 피해야 합니다. 이는 예기치 않은 부작용을 초래할 수 있습니다.
- 병렬 처리의 주의: 병렬 스트림을 사용할 때는 데이터 간의 의존성을 철저히 검토해야 합니다. 병렬 처리는 성능을 향상시킬 수 있지만, 잘못된 사용은 오히려 성능 저하를 초래할 수 있습니다.
7. 예제와 분석
지금까지 배운 스트림 API 개념들을 종합적으로 적용한 예제를 살펴보겠습니다.
종합 예제:
import java.util.*;
import java.util.stream.*;
public class StreamAdvancedExample {
public static void main(String[] args) {
List<String> names = Arrays.asList("Alice", "Bob", "Charlie", "David", "Eve");
// 필터링: 이름이 'A'로 시작하는 것만 필터링
List<String> filteredNames = names.stream()
.filter(name -> name.startsWith("A"))
.collect(Collectors.toList());
// 매핑: 이름을 대문자로 변환
List<String> upperCaseNames = filteredNames.stream()
.map(String::toUpperCase)
.collect(Collectors.toList());
// 정렬: 알파벳 순으로 정렬
List<String> sortedNames = upperCaseNames.stream()
.sorted()
.collect(Collectors.toList());
// 최종 결과 출력
sortedNames.forEach(System.out::println);
}
}
코드 분석:
- 'filter()'를 사용하여 이름이 'A'로 시작하는 요소만 필터링합니다.
- 'map()'을 사용하여 필터링된 이름을 대문자로 변환합니다.
- 'sorted()'를 사용하여 이름을 알파벳 순으로 정렬합니다.
- 최종적으로 'forEach()'를 사용하여 결과를 출력합니다.
8. 결론 및 추가 학습 자료
이번 글에서는 자바의 스트림 API에 대해 살펴보았습니다. 스트림 API는 데이터를 처리하는 데 매우 강력한 도구로, 선언형 프로그래밍 스타일을 지원하여 코드의 간결성과 가독성을 크게 향상시킵니다. 스트림을 사용하면 데이터를 필터링, 매핑, 정렬, 집계, 수집하는 작업을 쉽게 수행할 수 있으며, 병렬 처리도 간편하게 구현할 수 있습니다.
추가 학습 자료:
- 자바 공식 문서: [Oracle Java Documentation - Streams](https://docs.oracle.com/javase/tutorial/collections/streams/)
- 온라인 자바 튜토리얼: [W3Schools Java Streams](https://www.w3schools.com/java/java_stream.asp)
- 자바 코딩 연습 사이트: [GeeksforGeeks - Stream in Java](https://www.geeksforgeeks.org/streams-in-java/)
스트림 API는 자바의 중요한 기능 중 하나로, 다양한 데이터 처리 작업을 효율적으로 수행할 수 있습니다. 이번 기회를 통해 스트림 API의 개념과 활용 방법을 잘 이해하고, 실무에서 효과적으로 사용해보세요.
이제 자바의 스트림 API에 대해 자세히 이해하게 되었습니다. 다음 글에서는 자바의 또 다른 고급 기능에 대해 다루도록 하겠습니다. 자바의 더 깊은 이해를 위해 계속해서 학습해나가세요!