Collection이나 Iterator 같은 인터페이스를 이용해서 컬렉션을 다루는 방식을 표준화 했지만, 각 컬렉션 클래스에는 같은 기능의 메서드들이 중복해서 정의되어 있습니다.
List를 정렬할 때는 Collection.sort()를 사용해야하고, 배열을 정렬할 때는 Arrays.sort()를 사용해야 합니다.
JDK1.8 부터 다양한 데이터 소스를 표준화된 방법으로 다루기 위해 나온 것이 Stream 입니다.
스트림은 너무 양이 많아 그냥 쓸때마다 찾아와서 쓰는 용도로 작성했습니다.
1.스트림 만들기
2.중간연산(0~n번)
3.최종 연산(1번)
// 정렬된 결과의 새로운 List에 담아서 반환한다.
// 기존
String[] strArr = {"aaa", "bbb", "ccc"};
List<String> strList = Arrays.asList(strArr);
// 스트림 생성
Stream<String> strStream1 = strList.stream();
Stream<String> strStream2 = Arrays.stream(strArr);
// 스트림 출력
strStream1.sorted().forEach(System.out::println);
strStream2.sorted().forEach(System.out::println);
List<String> sortedList = strStream2.sorted().collect(Collectors.toList());
스트림은 일회용 입니다
스트림은 Iterator처럼 일회용입니다. Iterator로 컬렉션의 요소를 모두 읽고 나면 다시 사용할수 없는 것처럼 스트림도한번사용하면 닫혀서 다시 사용할 수 없습니다. 필요하다면 스트림을 다시 생성해야합니다.
strStream1.sorted().forEach(System.out.::println);
int numOfStr = strStream1.count(); //에러. 스트림이 이미 닫혔음.
스트림은 작업을 내부 반복으로 처리합니다.
스트림을 이용한 작업을 간결할 수 있는 비결중의 하나가 내부 반복입니다.
내부반복이라는 것은 반복문을메서드의 내부에 숨길 수 있다는 것을 의미합니다. forEach()는 스트림에 정의된 메서드 중의 하나로 매개변수에 대입된 람다식을 데이터 소스의 모든 요소에 적용합니다.
void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action); // 매개변수의 널 체크
for(T t : src) {
action.accept(T);
}
}
스트림 연산
중간연산 연산 결과가 스트림인 연산, 스트림에 연속해서 중간 연산할 수 있습니다
최종연산 연산 결과가 스트림이 아닌 연산,스티림의 요소를 소모하므로 단 한번만 가능합니다.
String[] strArr = {"dd", "aaa", "CC", "cc", "b"};
Stream<String> stream = Stream.of(strArr); // 문자열 배열이 소스인 스트림
Stream<String> filteredStream = stream.filter(); // 걸러내기(중간 연산)
Stream<String> distinctedStream = stream.distinct(); // 중복 제거(중간 연산)
Stream<String> sortedStream = stream.sort(); // 정렬 (중간 연산)
Stream<String> limitedStream = stream.limit(5); // 스트림 자르기 (중간 연산)
int total = stream.count(); // 요소 개수 세기 (최종 연산) - 스트림 닫힘 closed
스트림의 중간 연산 목록
중간연산 | 설명 |
Stream distinct() | 중복을제거 |
Stream filter (Predicate predicate) | 조건에 안 맞는요소제외 |
Stream limit (long maxSi ze) | 스트림의 일부를 낸다. |
Stre n skip(long n) | 스트림의 일부를 건너띈다. |
Stre n skip(long n) | 스트림의 소에 작업수행 |
Stream sorted() Stream sorted(Comparator comparator) |
스트림의 요소를 정렬한다 |
Stream map (Function mapper) DoubleStream mapToDouble (ToDoubleFunction mapper) IntStream mapTolnt (TolntFunction mapper) LongStream mapToLong (ToLongFunction mapper) Stream flatMap (Function> mapper) DoubleStream flatMapToDouble (Function m) IntStream flatMapTolnt (Function m) LongStream flatMapToLong (Function m) |
스트림의 요소를 변환한다. |
스트림의 최종 연산 목록
최종 연산 | 설명 |
void forEach (Consume r action ) void forEachOrdered(Consumer action) |
각 소에 정된 작업 수행 |
long count ( ) | 스트림의 의 개수 반환 |
Optional max (Comparator comparator) Optional min (Comparator compa rator) |
스트림의 최대 /최소값을 반환 |
Optional findAny ( ) Optional findFirst ( ) |
스트림의 요소 하나를 반환 |
boolean allMatch (Predicate p) boolean anyMatch (Predicate p) boolean noneMatch (Predi cate<> p) |
주어진 조건을 모든 요소가 만족시키는지, 만족시키지 않는지 확인 |
Object[] toArray() A[] toArray (IntFunction generator) |
스트림의 모든 요소를 배열로 반환 |
Optional reduce (BinaryOperator accumulator) T reduce (T identity, BinaryOperator accumulator) U reduce (U identity, BiFunction accumulator, BinaryOperator combiner) |
스트림의 요소를 하나씩 줄여가면서(리듀싱)계산 한다. |
R collect (Collector collector) R collect (Supplier supplier, BiConsumer accumulator, BiConsumer combiner) |
스트림의 요소를 수집한다. 주로 요소를 그룹화하거나 분할한 결과를 컬렉션에 담아 반환하는데 사용한다. |
지연된 연산
스트림 연산에 한 가지 중요한 점은 최종 연산이 수행되기 전까지 중간 연산이 수행되지 않는다는 것입니다.
Stream<Integer>와 IntStream
기본적으로 Stream<T>이지만 오토박싱&언박싱 으로 인한 비효율을 줄이기 위해 데이터 소스의 요소를 기본형으로 다루는 스트립 을 제공합니다 일반적으로 Stream<Integer>대신 IntStream 을 사용하는것이 더 효율적입니다.
병렬 스트림
스트림 만들기
Stream<T> Collection.stream()
Stream<T> Collection.stream()
// List로부터 스트림 생성
List<Integer> list = Arrays.asList(1,2,3,4,5);
Stream<Integer> intStream = list.stream(); // list를 소스로 하는 컬렉션 생성
배열
배열을 소스로 하는 스트림을 생성하는 메서드는 다음과 같이 Stream과 Arrays에 static메서드로 정의되어 있습니다.
Stream<T> Stream.of(T... values) // 가변 인자
Stream<T> Stream.of(T[])
Stream<T> Arrays.stream(T[])
Stream<T> Arrays.stream(T[] array, int startInclusive, int endExclusive)
// 문자열 스트림 생성
Stream<String> strStream = Stream.of("a","b","c"); // 가변인자
Stream<String> strStream = Stream.of(new String[]{"a","b","c"});
Stream<String> strStream = Arrays.stream(new String[]{"a","b","c"});
Stream<String> strStream = Arrays.stream(new String[]{"a","b","c"}, 0, 3);
// int, long, double과 같은 기본형 배열을 소스로 하는 스트림 생성
IntStream IntStream.of(int...values)
IntStream IntStream.of(int[])
IntStream Arrays.stream(int[])
IntStream Arrays.stream(int[] array, int startInclusive, endExclusive)
특정 범위의 정수
IntStream intStream = IntStream.range(1, 5); // 1,2,3,4
IntStream intStream = IntStream.rangeClosed(1, 5) // 1,2,3,4,5
임의의 수
난수를 생성하는 Random클래스에는 인스턴스 메서드들이 포함이 됩니다.
//해당 타입의 난수들로 이루어진 스트림 반환하는 메서드들
IntStream ints()
LongStream longs()
DoubleStream doubles()
//위의 메서드들이 반환하는 스트림은 크기가 정해지지 않은 무한 스트림(infinite stream)이므로
//limit()도 같이 사용해서 스트림의 크기를 제한해 주어야 한다. limit()은 스트림의 개수를 지정하는데 사용된다.
IntStream intStream = new Random().ints(); // 무한스트림
intStream.limit(5).forEach(System.out::println); // 5개의 요소만 출력
//매개변수로 스트림의 크기를 지정하면 limit()을 사용하지 않아도 된다.
IntStream ints(long streamSize)
LongStream longs(long streamSize)
DoubleStream doubles(long streamSize)
//지정된 범위의 난수를 발생시키는 스트림을 얻는 메서드들(end는 범위에 미포함)
IntStream ints(int begin, int end)
LongStream longs(long begin, long end)
DoubleStream doubles(double begin, double end)
IntStream ints(long streamSize, int begin, int end)
LongStream longs(long streamSize, long begin, long end)
DoubleStream doubles(long streamSize, double begin, double end)
람다식 - iterate(), generate()
Stream 클래스의 iterate()와generate()는 람다식을 매개변수로 받아서, 이 람다식에 의해 계산되는 값들을 요소로 하는
무한 스트림을 생성합니다
static <T> Stream<T> iterate(T seed, UnaryOperator<T> f)
static <T> Stream<T> generate(Supplier<T> s)
iterate()는 씨앗값(seed)으로 지정된 값부터 시작해서 람다식 f에 의해 계산된 결과를 다시 seed값으로 계산을 반복합니다
generate()는 iterate()와 달리 이전 결과를 이용해서 다음 요소를 계산하지 않습니다
terate()와 generate()에 의해 생성된 스트림은 기본형 스트림 타입의 참조변수로 다룰 수 없습니다.
빈 스트림
요소하나도 없는 비어있는 스트림을 생성할 수있습니다.
Stream emptyStream = Stream.empty(); // empty()는 빈 스트림을 생성해서 반환한다.
long count = emptyStream.count(); // count의 값은 0
두 스트림의 연결
Stream의 static 메서드인 concat()을 사용해서 두 스트림을 하나로 연결할 수 있습니다
두스트림은 같은 타입이어야 합니다.
String[] str1 = {"123", "456", "789"}
String[] str2 = {"ABC", "abc", "DEF"}
Stream<String> strs1 = Stream.of(str1);
Stream<String> strs2 = Stream.of(str2);
Stream<String> strs3 = Stream.concat(strs1, strs2); // 두 스트림을 하나로 연결
스트림의 중간연산
스트림 자르기
Stream skip(long n) // 처음 n개의 요소 건너뛰기
Strema limit(long maxSize) // 스트림의 요소를 maxSize개로 제한
IntStream exampleStream = IntStream.rangeClosed(1, 10); // 1~10의 요소를 가진 스트림
exampleStream.skip(3).limit(5).forEach(System.out::print); // 45678
스트림 요소 걸러내기
distinct()는 스트림에서 중복된 요소들 제거
filter()는 주어진 조건(Predicate)에 맞지 않는 요소를 걸러냅니다
// distinct()
IntStream exampleStream = IntStream.of(1,2,2,3,3,3,4,5,5,6);
exampleStream.distinct().forEach(System.out::print); // 123456
// filter()
IntStream example2Stream = IntStream.rangeClosed(1, 10);
example2Stream.filter(i -> i%2 ==0).forEach(System.out::print); // 246810
// filter()를 다른 조건으로 여러 번 사용. 두 문장의 결과는 같다.
example2Stream.filter(i -> i%2!=0 && i%3!=0).forEach(System.out::print); //157
example2Stream.filter(i -> i%2!=0).filter(i -> i%3!=0).forEach(System.out::print);
스트림 정렬
스트림을 정렬할 때는 sorted()를 사용하면 됩니다.
tream<String> strStream = Stream.of("dd","aaa","CC","cc","b");
strStream.sorted().forEach(System.out::print); // CCaaabccdd
// 기본정렬 역순
strStream.sorted(Comparator.reverseOrder());
// 대소문자 구분 없이
strStream.sorted(String.CASE_INSESITIVE_ORDER)
// 길이 순 정렬
strStream.sorted(Comparator.comparing(String::length))
//정렬 조건을 추가할 때는 thenComparing() 사용
studentStream.sorted(Comparator.comparing(Student::getBan)
.thenComparing(Student::getTotalScore)
.thenComparing(Student::getName)
.forEach(System.out::println);
스트림 변환
스트림의 요소에 저장된 값 중에서 원하는 필드만 뽑아내거나, 특정 형태로 변환해야 할 때
map() 사용합니다.
/ map()으로 Stream<File>을 Stream<String>으로 변환하기
Stream<File> fileStream = Stream.of(new File("Ex1.java"), new File("Ex2.java"));
Stream<String> filenameStream = fileStream.map(File::getName);
filenameStream.forEach(System.out::println);
// map() 여러 번 사용
fileStream.map(File::getName) // Stream<File> -> Stream<String>
.filter(s -> s.indexOf('.')!= -1) // 확장자 없는 것 제외
.map(s -> s.substring(s.indexOf('.')+1)) // Stream<String> -> Stream<String>
.map(String::toUpperCase) // 모두 대문자로 변환
.distinct()
.forEach(System.out::print);
스트림 조회
연산과 연산 사이에 올바르게 처리되었는지 확인하려면 peek()사용합니다
forEach()와 달리 스트림의 요소를 소모하지 않으므로 연산 사이에 여러 번 사용 가능
fileStream.map(File::getName)
.filter(s -> s.indexOf('.') != -1) //확장자 없는 것 제외
.peek(s -> System.out.printf("filename=%s%n", s)) //파일명 출력
.map(s -> s.substring(s.indexOf('.')+1)) //확장자만 추출
.peek(s -> System.out.printf("extension=%s%n", s)) //확장자 출력
.forEach(System.out::println);
Stream 스트림을 기본형 스트림으로 변환
map()은 연산자의 결과로 Stream 타입의 스트림을 반환하는데,
스트림의 요소를 숫자로 반환하는 경우 InStream과 같은 기본형 스트림으로 변환하는 것이 더 유용할 수 있습니다.
DoubleStream mapToDouble(ToDoubleFunction<? super T> mapper)
IntStream mapToInt(ToIntFunction<? super T> mapper)
LongStream mapToLong(ToLongFunction<? super T> mapper)
// 스트림에 포함된 모든 학생의 성적을 합산하기 위해 map()으로 새로운 스트림 생성
Stream<Integer> studentScoreStream = studentStream.map(Student::getTotalScore);
// 애초에 Stream<Integer>가 아닌 IntStream 타입의 스트림 생성하기
// 성적을 더할 때 Integer를 int로 변환할 필요 없어서 더 효율적이다.
IntStream studentScoreStream = studentStream.mapToInt(Student::getTotalScore);
int allTotalScore = studentScoreStream.sum();
//mapToObj() 사용해서 IntStream을 Stream<String>으로 변환
//로또번호 생성기
IntStream intStream = new Random().ints(1,46); // 1~45사이의 정수
Stream<String> lottoStream = intStream.distinct().limit(6).sorted()
.mapToObj(i->i+","); // 정수를 문자열로 변환
lottoStream.forEach(System.out::print); // 12,14,23,29,45
Stream를 Stream로 변환
flatMap() 사용
// 요소가 문자열 배열(String[])인 스트림
Stream<String[]> strArrStrm = Stream.of(
new String[]{"abc", "def", "ghi"},
new String[]{"ABC", "DEF"m "JKLMN"}
);
// Stream<String[]>을 map(Arrays::stream)으로 변환한 결과는 Stream<String>이 아닌 Stream<Stream<String>>.
// 즉 스트림의 스트림이다.
Stream<Stream<String>> strStrStrm = strArrStrm.map(Arrays::stream);
//map() 대신 flatMap() 사용하면 각 배열이 하나의 스트림의 요소가 된다.
Stream<String> strStrm = strArrStrm.flatMap(Arrays::stream);
Optional
Optional은 지네릭 클래스로 ‘T타입의 객체’를 감싸는 래퍼 클래스입니다
그래서 Optional타입의 객체에는 모든 타입의 참조변수를 담을 수 있습니다.
최종 연산의 결과를 그냥 반환하는 것이 아니라 Optional 객체에 담아서 반환하면,
반환된 결과가 null인지 매번 if문으로 체크하는 대신 Optional에 정의된 메서드를 통해 간단히 처리할 수 있습니다.
Optional 객체 생성하기
of() 또는 ofNullable() 사용
//참조변수의 값이 null일 가능성이 있으면 of()대신 ofNullable() 사용
Optional<String> optVal = Optional.of(null); //NullPointerException 발생
Optional<String> optVal = Optional.ofNullable(null); // OK
//Optional<T> 타입의 참조변수를 기본값으로 초기화할 때 empty() 사용
//null로 초기화할 수 있지만 empty()로 초기화하는 것이 바람직
Optional<String> optVal = null;
Optional<String> optVal = Optional.empty(); // 빈 객체로 초기화
Optional 객체의 값 가져오기
get(), orElse()
Optional<String> optVal = Optional.of("abc");
String str1 = optVal.get(); // optVal에 저장된 값을 반환. null이면 예외발생
String str2 = optVal.orElse(""); // optVal에 저장된 값이 null이면 ""를 반환
String str3 = optVal.orElseGet(String::new); // null을 대체할 값을 반환하는 람다식 지정
String str4 = optVal.orElseThrow(NullPointerException::new); // null일 때 지정된 예외를 발생
isPresent()는 Optional 객체의 값이 null이면 false를, 아니면 true 반환
ifPresent(Consumer block)는 값이 있으면 주어진 람다식 실행, 없으면 아무일 안합니다.
//조건문
if(str!=null) {
System.out.println(str);
}
//위의 조건문을 isPresent()로 구현
if(Optional.ofNullable(str).isPresent()) {
System.out.println(str);
}
//위의 조건문을 ifPresent()로 구현
Optional.ofNullable(str).ifPresent(System.out::println);
스트림의 최종 연산
최종 연산은 스트림의 요소를 소모해서 결과를 만든다. 그래서 최종 연산후에는 스트림이 닫혀서 더이 상 사용할 수 없습니다.
forEach()
반환 타입이 void이므로 스트림의 요소를 출력하는 용도로 많이 사용합니다.
조건검사
allMatch(), anyMatch(), noneMatch(), findFirst(), findAny()
boolean allMatch (Predicate<? super T> predicate)
boolean anyMatch (Predicate<? super T> predicate)
boolean noneMatch (Predicate<? super T> predicate)
// 학생성적 스트림에서 총점이 낙제점 이하인 학생 확인
boolean noFailed = stuStream.anyMatch(s->s.getTotalScore()<=100);
// 스트림 요소 중 조건에 일치하는 첫 번째 것 반환
Optional<Student> stu = stuStream.filter(s->s.getTotalScore()<=100).findFirst();
// 병렬스트림의 경우 findFirst() 대신 findAny() 사용
Optional<Student> stu = parallelStream.filter(s->s.getTotalScore()<=100).findAny();
통계
count(), sum(), average(), max(), min()
// 기본형 스트림이 아닌 경우 통계 관련 메서드는 아래 3개 뿐이다.
// 기본형 스트림의 min(), max()와 달리 매개변수로 Comparator가 필요하다.
long count()
Optional<T> max(Comparator<? super T> comparator)
Optional<T> min(Comparator<? super T> comparator)
reduce()
reduce()는 스트림의 요소를 줄여나가면서 연산을 수행하고 최종결과를 반환합니다.
그래서 매개변수의 타입이 BinaryOperator
처음 두 요소를 가지고 연산한 결과로 그 다음 요소와 연산합니다.
모든 스트림의 요소를 소모하게 되면 그 결과를 반환합니다.
Optional<T> reduce(BinaryOperator<T> accumulator)
reduce()의 사용방법은 초기값과 어떤 연산(BinaryOperator)으로 스트림의 요소를 줄여나갈 것인지만 결정하면 됩니다.
collect()
스트림의 요소를 수집하는 최종 연산
collect()가 스트림의 요소를 수집하기 위한 수집 방법이 정의된 것이 collector.
collector는 Collector인터페이스를 구현한 것.
collect() 스트림의 최종 연산. 매개변수로 collector가 필요하다.
Collector 인터페이스. collector는 이 인터페이스를 구현해야한다.
Collectors 클래스. static 메서드로 미리 작성된 collector를 제공한다.
collect()의 매개변수 타입은 Collector인데. 매개변수가 Collector를 구현한 클래스의 객체이어야 한다는 뜻.
그리고 collect()는 이 객체에 구현된 방법대로 스트림의 요소를 수집한다.
스트림을 컬렉션과 배열로 변환
toList(), toSet(), toMap(), toCollection(), toArray()
// 스트림의 모든 요소를 컬렉션에 수집하려면, Collectors클래스의 toList()와 같은 메서드를 사용하면 된다.
List<String> names = stuStream.map(Student::getName)
.collect(Collectors.toList());
// List나 Set이 아닌 특정 컬렉션을 지정하려면,
// toCollection()에 해당 컬렉션의 생성자 참조를 매개변수로 넣어주면 된다.
ArrayList<String> list = names.stream()
.collect(Collectors.toCollection(ArrayList::new));
// Map은 객체의 어떤 필드를 키와 값으로 사용하지 지정해야 한다.
// 요소의 타입이 Person인 스트림에서 사람의 주민번호(regId)를 키로 하고, 값으로 Person 객체를 그대로 저장
Map<String, Person> map = personStream.collect(Collectors.toMap(p->p.getRegId(), p->p)
// 스트림에 저장된 요소들을 T[] 타입의 배열로 변환하려면 toArray() 사용
// 단, 해당 타입의 생성자 참조를 매개변수로 지정해줘야 한다. 지정하지 않으면 반환되는 배열의 타입은 Object[]
Student[] stuNames = studentStream.toArray(Student[]::new); // OK
Student[] stuNames = studentStream.toArray(); // 에러
Object[] stuNames = studentStream.toArray(); // OK
통계
counting(), summingInt(), averagingInt(), maxBy(), minBy()
// 기본형 스트림이 아닌 경우 통계 관련 메서드는 아래 3개 뿐이다.
// 기본형 스트림의 min(), max()와 달리 매개변수로 Comparator가 필요하다.
long count()
Optional<T> max(Comparator<? super T> comparator)
Optional<T> min(Comparator<? super T> comparator)
문자열 결합
joining()
스트림의 요소가 String이나 StringBuffer처럼 CharSequence의 자손인 경우에만 결합 가능하므로
스트림의 요소가 문자열이 아닌 경우에는 먼저 map()을 이용해서 스트림의 요소를 문자열로 변환해야 합니다.
String studentNames = stuStream.map(Student::getName).collect(joining());
String studentNames = stuStream.map(Student::getName).collect(joining(","));
String studentNames = stuStream.map(Student::getName).collect(joining(",", "[", "]"));
// 만약 map()없이 스트림에 바로 joining()하면, 스트림의 요소에 toString()을 호출한 결과를 결합한다.
String studentInfo = stuStream.collect(joining(","));
그룹화와 분할
groupingBy(), partitioningBy()
그룹화는 스트림의 요소를 특정 기준으로 그룹화하는 것입니다.
분할은 스트림의 요소를 두 가지, 지정된 조건에 일치하는 그룹과 아닌 그룹으로 분할하는 것입니다.
groupingBy()는 스트림의 요소를 Function으로, partitioningBy()는 Predicate로 분류
보통 스트림의 두 개의 그룹으로 나눠야 하면 partitioningBy()쓰는 것이 더 빠르고, 그 외에는 groupingBy()를 쓰면 됩니다.
그룹화와 분할의 결과는 Map에 담겨 반환됩니다.
partitioningBy()에 의한 분류
// 1. 기본 분할
Map<Boolean, List<Student>> stuBySex
= stuStream.collect(partitioningBy(Student::isMale)); // 학생들을 성별로 분할
List<Student> maleStudent = stuBySex.get(true); // Map에서 남학생 목록을 얻는다.
List<Student> femaleStudent = stuBySex.get(false); // 여학생 목록
// 2. 기본 분할 + 통계 정보
Map<Boolean, Long> stuNumBySex = stuStream.collect(partitioningBy(Student::isMale, counting()));
System.out.println(stuNumBySex.get(true)); // 8 (남학생수)
System.out.println(stuNumBySex.get(false)); // 10 (여학생수)
// 남학생 1등 구하기, mapBy()의 반환타입은 Optional<Student>
Map<Boolean, Optional<Student>> topScoreBySex
= stuStream.collect(partitioningBy(Student::isMale, maxBy(comparingInt(Student::getScore))));
System.out.println(topScoreBySex.get(true)); // Optional{[남일등, 남, 1, 1, 300]}
// mapBy()의 반환타입이 Optional<Student>가 아닌 Student를 반환 결과로 얻으려면,
// collectiongAndThen()과 Optional::get 함께 사용
Map<Boolean, student> topScoreBySex
= stuStream.collect(
partitioningBy(
Student::isMale, collectingAndThen(
maxBy(comparingInt(Student::getScore))
, Optional::get)));
// 3. 이중 분할
Map<Boolean, Map<Boolean, List<Student>>> failedStuBySex
= stuStream.collect(
partitioningBy(Student::isMale, partitioningBy(s->s.getScore()<150)));
List<Student> failedMaleStu = failedStuBySex.get(true).get(true);
groupingBy()에 의한 분류
groupingBy()로 그룹화하면 기본적으로 List에 담는다.
// 1. 학생 스트림을 반 별로 그룹지어 Map에 저장
Map<Integer, List<Student>> stuByBan
= stuStream.collect(groupingBy(Student::getBan, toList())); //toList()생략가능
Map<Integer, HashSet<Student>> stuByHak
= stuStream.collect(groupingBy(Student::getHak, toCollection(HashSet::new)));
// 2. 학생 스트림을 성적의 등급(Student.Level)으로 그룹화
Map<Student.Level, Long> stuByLevel
= stuStream.collect(
groupingBy(s-> { if(s.getScore()>=200) return Student.Level.HIGH;
else if(s.getScore()>=100) return Student.Level.MID;
else return Student.Level.LOW;
}, counting()));
// 3. groupingBy() 다중 사용하기.
// 학년별로 그룹화하고 다시 반별로 그룹화
Map<Integer, Map<Integer, List<Student>>> stuByHakAndBan
= stuStream.collect(groupingBy(Student::getHak,
groupingBy(Student::getBan)));
// 4. 각 반별 1등 추출
Map<Integer, Map<Integer, Student>> topStuByHakAndBan
= stuStream.collect(groupingBy(Student::getHak,
groupingBy(Student::getBan,
collectingAndThen(
maxBy(comparingInt(Student::getScore)),
Optional::get))));
// 5. 학년별, 반별 그룹화한 후에 성적그룹으로 변환하여 Set에 저장
Map<Integer, Map<Integer, Set<Student.Level>>> stuByHakAndBan
= stuStream.collect(groupingBy(Student::getHak,
groupingBy(Student::getBan,
mapping(s-> {
if(s.getScore()>=200) return Student.Level.HIGH;
else if(s.getScore()>=100) return Student.Level.MID;
else return Student.Level.LOW;
} , toSet()))));