1. ์คํธ๋ฆผ(Stream)?
๋ง์ ์์ ๋ฐ์ดํฐ๋ฅผ ๋ค๋ฃฐ ๋ for๋ฌธ์ด๋ Iterator๋ฅผ ์ฌ์ฉํ์ฌ ์ฝ๋๋ฅผ ์์ฑํด์๋ค. ๋ํ ๋ฐ์ดํฐ์ ํํ๋ง๋ค ๋ค๋ฅธ ๋ฐฉ์์ผ๋ก ๋ค๋ค์ผ ํ๋ค. ์๋ฅผ๋ค์ด, ๋ฐ์ดํฐ์ ์ ๋ ฌ์ ์๋ก ๋ค์๋ฉด ์ปฌ๋ ์ ํด๋์ค ์ค List๋ฅผ ์ ๋ ฌํ ๋์๋ Collections.sort()๋ฅผ, ๋ฐฐ์ด์ ์ ๋ ฌํ ๋๋ Arrays.sort()๋ฅผ ์ฌ์ฉํด์ผํ๋ค. ๊ฐ์ ๊ธฐ๋ฅ์ด๊ณ ๊ฐ์ ๋ฉ์๋์์๋ ๋ค๋ฅด๊ฒ ํธ์ถ์ ํด์ผ๋ง ํ๋ค.
์ด๋ฌํ ๋ฌธ์ ์ ๋ค์ ํด๊ฒฐํ๊ธฐ ์ํด ๋ง๋ ๊ฒ์ด ๋ฐ๋ก ์๋ฐ 8๋ถํฐ ๋ฑ์ฅํ ์คํธ๋ฆผ(Stream)์ด๋ค. ์ด๋ค ๋ฐ์ดํฐ์ด๋ ๊ฐ์ ๋ฐฉ์์ผ๋ก ๋ค๋ฃฐ ์ ์๊ฒ ๋ง๋ ๊ฒ์ด๋ค. ์คํธ๋ฆผ์ ์ด์ฉํ๋ฉด, ๋ฐฐ์ด์ด๋ ์ปฌ๋ ์ ๋ฟ๋ง ์๋๋ผ ํ์ผ์ ์ ์ฅ๋ ๋ฐ์ดํฐ๋ ๋ชจ๋ ๊ฐ์ ๋ฐฉ์์ผ๋ก ๋ค๋ฃฐ ์ ์๊ธฐ ๋๋ฌธ์ ์ฝ๋์ ์ฌ์ฌ์ฉ์ฑ์ด ๋์์ง๋ค. ์์ ๋ฅผ ๋ณด๋ฉด ํ ๋์ ์ดํดํ ์ ์๋ค.
String[] strArr = { "aaa", "bbb", "ccc" };
List strList = Arrays.asList(strArr);
Stream strStream1 = strList.stream(); // ์คํธ๋ฆผ์ ์์ฑ
Stream strStream2 = Arrays.stream(strArr); // ์คํธ๋ฆผ์ ์์ฑ
strStream1.sorted().forEach(System.out::println);
strStream2.sorted().forEach(System.out::println);
// ์ด์ ์ฝ๋
/*
Arrays.sort(strArr);
Collections.sort(strList);
for (String str : strArr)
System.out.println(str);
for (String str : strList)
System.out.println(str);
*/
์คํธ๋ฆผ์ ์ฌ์ฉํ ์ฝ๋๊ฐ ๊ฐ๊ฒฐํ๊ณ ์ดํดํ๊ธฐ ์ฌ์ฐ๋ฉด์ ์ฌ์ฌ์ฉ์ฑ์ด ๋๋ค๋ ๊ฒ์ ์ ์ ์๋ค.
์ฌ๊ธฐ์ ์คํธ๋ฆผ์ ์๋ณธ๋ฐ์ดํฐ๋ฅผ ๋ณ๊ฒฝํ์ง ์๋๋ค.
์๋ณธ๋ฐ์ดํฐ์์ stream์ ์์ฑํ์ฌ ์ฌ์ฉํ๊ธฐ ๋๋ฌธ์ด๋ค. ์คํธ๋ฆผ์ ํตํด ์ ๋ ฌํ ๊ฒฐ๊ณผ๋ฅผ ์ฌ์ฉํ๊ณ ์ถ๋ค๋ฉด ์ปฌ๋ ์ ์ด๋ ๋ฐฐ์ด์ ๋ด์์ ๋ฐํํ ์๋ ์๋ค.
์คํธ๋ฆผ์ ์ผํ์ฉ์ด๋ค.
์คํธ๋ฆผ์ Iterator์ฒ๋ผ ์ผํ์ฉ์ด๋ค. Iterator๋ก ์ปฌ๋ ์ ์ ๋ชจ๋ ์์๋ฅผ ์ฝ๊ณ ๋ ํ์ ๋ค์ ์ฌ์ฉํ ์ ์๋ ๊ฒ์ฒ๋ผ, ์คํธ๋ฆผ๋ ํ๋ฒ ์ฌ์ฉํ๋ฉด ๋ซํ๊ธฐ ๋๋ฌธ์ ์ฌ์ฌ์ฉ์ด ๋ถ๊ฐ๋ฅํ๋ค.
์คํธ๋ฆผ์ ์์ ์ ๋ด๋ถ ๋ฐ๋ณต์ผ๋ก ์ฒ๋ฆฌํ๋ค.
๋ด๋ถ ๋ฐ๋ณต์ด๋, ๋ฐ๋ณต๋ฌธ์ ๋ฉ์๋์ ๋ด๋ถ์ ๋ฃ์๋ค๋๊ฒ์ ์๋ฏธํ๋ค. ์์ ์์ ์ฝ๋์์ forEach()๋ ์คํธ๋ฆผ์ ์ ์๋ ๋ฉ์๋ ์ค ํ๋๋ก ๋งค๊ฐ๋ณ์์ ๋์ ๋ ๋๋ค์์ ๋ชจ๋ ๋ฐ์ดํฐ์ ๋ํด for๋ฌธ ๋๋ฆฌ๋ ๊ฒ์ด๋ผ๊ณ ์๊ฐํ๋ฉด
void forEach(Consumer action) {
Objects.requireNonNull(action); // ๋งค๊ฐ๋ณ์์ ๋ ์ฒดํฌ
for(T t : src) { // ๋ด๋ถ๋ฐ๋ณต
action.accept(T);
}
}
์คํธ๋ฆผ์ ๋ค์ํ ์ฐ์ฐ์ ์ ๊ณตํ์ฌ ๋ณต์กํ ์์ ์ ๊ฐ๋จํ๊ฒ ์ฒ๋ฆฌํ ์ ์๋ค. ๋ง์น DB์ ์ฟผ๋ฆฌ์ ๋น์ทํ๋ค. ์ฌ๊ธฐ์ ์ฐ์ฐ์ด๋ ์คํธ๋ฆผ์์ ์ ์๋ ๋ฉ์๋ ์ค์ ๋ฐ์ดํฐ๋ฅผ ๋ค๋ฃจ๋ ๋ฉ์๋๋ฅผ ์๋ฏธํ๋ค. (ํ๋ง๋๋ก ๋ฉ์๋)
์คํธ๋ฆผ์ด ์ ๊ณตํ๋ ์ฐ์ฐ์ ์ค๊ฐ์ฐ์ฐ๊ณผ ์ต์ข ์ฐ์ฐ์ผ๋ก ๋ถ๋ฅ๋๋ค.
์ค๊ฐ์ฐ์ฐ์ ์ฐ์ฐ ๊ฒฐ๊ณผ๋ฅผ ์คํธ๋ฆผ์ผ๋ก ๋ฐํํ๊ธฐ ๋๋ฌธ์ ์ฐ์ํด์ ์ฐ๊ฒฐํ ์ ์๋ค. ๊ทธ์ ๋ฐํด ์ต์ข ์ฐ์ฐ์ ์คํธ๋ฆผ์ ์์๋ฅผ ์๋ชจํ๋ฉด์ ์ฐ์ฐ์ ์ํํ๊ธฐ ๋๋ฌธ์ ๋จ ํ๋ฒ๋ง ์ฐ์ฐ์ด ๊ฐ๋ฅํ๋ค.
stream.distinct().limit(5).sorted().forEach(System.out::println)
Stream์ ์ ์๋ ์ฐ์ฐ์ ์ฑ ์ ์ฐธ๊ณ ํ์.
์ค๊ฐ์ฐ์ฐ์ map()๊ณผ flatMap(), ์ต์ข ์ฐ์ฐ์ reduce()์ collect()๊ฐ ํต์ฌ์ด๋ค. ๋๋จธ์ง๋ ์ดํดํ๊ธฐ ์ฝ๊ณ ์ฌ์ฉ๋ฒ๋ ๊ฐ๋จํ๋ค.
์ง์ฐ๋ ์ฐ์ฐ
์คํธ๋ฆผ ์ฐ์ฐ์์ ์ค์ํ ์ ์ ์ต์ข ์ฐ์ฐ์ด ์ํ๋๊ธฐ ์ ๊น์ง๋ ์ค๊ฐ ์ฐ์ฐ์ด ์ํ๋์ง ์๋๋ค. ์คํธ๋ฆผ์ ๋ํด distinct()๋ sort()๊ฐ์ ์ค๊ฐ ์ฐ์ฐ์ ํธ์ถํด๋ ์ฆ๊ฐ์ ์ธ ์ฐ์ฐ์ด ์ํ๋๋ ๊ฒ์ด ์๋๋ค. ์ต์ข ์ฐ์ฐ์ด ํธ์ถ๋์ด์ผ ์ค๊ฐ ์ฐ์ฐ์ ๊ฑฐ์ณ ์ต์ข ์ฐ์ฐ์ด ์ํ๋๋ค.
Stream<Integer>์ IntStream
์คํธ๋ฆผ์ ๊ธฐ๋ณธ์ ์ผ๋ก Stream<T>์ด์ง๋ง, ์คํ ๋ฐ์&์ธ๋ฐ์ฑ์ผ๋ก ์ธํ ๋นํจ์จ์ ์ค์ด๊ธฐ ์ํด ๋ฐ์ดํฐ๋ฅผ ๊ธฐ๋ณธํ์ผ๋ก ๋ค๋ฃจ๋ ์คํธ๋ฆผ, IntStream, LongStream, DoubleStream์ด ์ ๊ณต๋๋ค. ์ผ๋ฐ์ ์ผ๋ก Stream<Integer> ๋์ IntStream์ ์ฌ์ฉํ๋ ๊ฒ์ด ๋ ํจ์จ์ ์ด๋ผ๊ณ ํ๋ค. ๋ํ IntStream์๋ intํ์ ์ ๊ฐ์ ๋ค๋ฃจ๋๋ฐ ์ ์ฉํ ๋ฉ์๋๋ค์ด ํฌํจ๋์ด ์๋ค๊ณ ํ๋ค.
๋ณ๋ ฌ ์คํธ๋ฆผ
์คํธ๋ฆผ์ ์ฅ์ ์ผ๋ก ๋ณ๋ ฌ์ฒ๋ฆฌ๊ฐ ์ฝ๋ค๋ ๊ฒ์ด๋ค. parallel()์ด๋ผ๋ ๋ฉ์๋๋ฅผ ํธ์ถํด์ ๋ณ๋ ฌ๋ก ์ฐ์ฐ์ ์ํํ๋๋ก ํ๋ฉด๋๋ค. ๊ทธ์ ๋ฐ๋๋ก ์ง๋ ฌ ์คํธ๋ฆผ์ sequential()์ ํธ์ถํ๋ฉด ๋๋๋ฐ ๋ชจ๋ ์คํธ๋ฆผ์ ๊ธฐ๋ณธ์ ์ผ๋ก ์ง๋ ฌ ์คํธ๋ฆผ์ด๊ธฐ ๋๋ฌธ์ sequential()์ ๊ตณ์ด ํธ์ถํ ํ์ ์๋ค.
2. ์คํธ๋ฆผ ๋ง๋ค๊ธฐ
์คํธ๋ฆผ์ผ๋ก ์์ ํ๊ธฐ ์ํด์ ์คํธ๋ฆผ๋ถํฐ ์์ฑํ์. ์คํธ๋ฆผ์ด ๋ฝ์๋ผ ์ ์๋ ๋์์ ๋ง๊ธฐ ๋๋ฌธ์ ๋์์ ๋ฐ๋ผ ์ ๋ฆฌํด๋ณด์.
์ปฌ๋ ์
๋ฐฐ์ด
ํน์ ๋ฒ์์ ์ ์
์์์ ์
๋๋ค์
ํ์ผ
๋น ์คํธ๋ฆผ
๋ ์คํธ๋ฆผ์ ์ฐ๊ฒฐ
์ปฌ๋ ์
์ปฌ๋ ์ ์ Collection ํด๋์ค์ strearm()์ด ์ ์๋์ด ์๋ค. ๋ฐ๋ผ์ Collection์ ์์๋ฐ์ List์ Set์ ๊ตฌํํ ์ปฌ๋ ์ ํด๋์ค๋ค์ ๋ชจ๋ ์ด ๋ฉ์๋๋ก ์คํธ๋ฆผ์ ์์ฑํ ์ ์๋ค. ์๋ฅผ ๋ค์ด List๋ก๋ถํฐ ์คํธ๋ฆผ์ ์์ฑํด๋ณด์.
List list = Arrays.asList(1,2,3,4,5);
Stream intStream = list.stream();
intStream.forEach(System.out::println);
intStream.forEach(System.out::println); // ์๋ฌ ๋ฐ์. ์คํธ๋ฆผ์ด ์ด๋ฏธ ๋ซํ.
[์คํ๊ฒฐ๊ณผ]
1
2
3
4
5
java.lang.IllegalStateException:
stream has already been operated upon or closed
ํ ๊ฐ์ง ์ฃผ์ํ ์ ์ forEach()๋ฅผ jstl์ forEach์ ๊ฐ์ด ์๊ฐํ๋ฉด ๋๋๋ฐ ์คํธ๋ฆผ์ ์์๋ฅผ ์๋ชจํ๋ฉด์ ์์ ์ ์ํํ๊ธฐ ๋๋ฌธ์ ํ ๋ฒ ์ํ๋ ๋ค์๋ ๊ฐ์ ์คํธ๋ฆผ์ ๋ forEach()๋ฅผ ํธ์ถํ ์ ์๋ค. ๊ทธ๋์ ์คํธ๋ฆผ์ ์์๋ฅผ ํ๋ฒ ๋ ์ถ๋ ฅํ๋ ค๋ฉด ์คํธ๋ฆผ์ ์๋ก ์์ฑํด์ผํ๋ค.
๋ฐฐ์ด
๋ฐฐ์ด๋ก ์คํธ๋ฆผ์ ์์ฑํ๋ ๋ฉ์๋๋ ๋ค์๊ณผ ๊ฐ์ด Stream๊ณผ Arrays์ static๋ฉ์๋๋ก ์ ์๋์ด ์๋ค.
Stream.of(T... values) // ๊ฐ๋ณ์ธ์
Stream.of(T[])
Arrays.stream(T[])
Arrays.stream(T[] array, int startInclusive, int endExclusive)
์๋ฅผ๋ค๋ฉด ๋ค์๊ณผ ๊ฐ๋ค.
Stream strStream = Stream.of("a", "b", "c"); // ๊ฐ๋ณ์ธ์
Stream strStream = Stream.of(new String[] {"a", "b", "c"});
Stream strStream = Arrays.stream(new String[] {"a", "b", "c"});
Stream strStream = Arrays.stream(new String[] {"a", "b", "c"}, 1, 3)
๊ทธ๋ฆฌ๊ณ int, long, double๊ณผ ๊ฐ์ ๊ธฐ๋ณธํ ๋ฐฐ์ด์ ๋ฐ์ดํฐ๋กํ๋ ์คํธ๋ฆผ์ ์์ฑํ๋ ๋ฉ์๋๋ ์๋ค.
IntStream IntStream.of(int... values)
IntStream IntStream.of(int[])
IntStream Arrays.stream(int[])
IntStream Arrays.stream(int[] array, int startInclusive, int endExclusive)
ํน์ ๋ฒ์์ ์ ์
์์์๋ ์ธ๊ธํ์ง๋ง ๊ธฐ๋ณธํ ๋ฐ์ดํฐ๋ฅผ ๋ค๋ฃจ๋ ์คํธ๋ฆผ์ ๊ธฐ๋ณธํ ๋ฐ์ดํฐ๋ฅผ ์ข ๋ ์ ์ฉํ๊ฒ ๋ค๋ฃฐ ์ ์๋ ๋ฉ์๋๊ฐ ์๋ค๊ณ ํ๋๋ฐ, IntStream๊ณผ LongStream์ ๋ค์๊ณผ ๊ฐ์ด ์ง์ ๋ ๋ฒ์์ ์ฐ์๋ ์ ์๋ฅผ ์คํธ๋ฆผ์ผ๋ก ์์ฑํด์ ๋ฐํํ๋ range()์ rangeClosed()๋ฉ์๋๋ฅผ ๊ฐ์ง๊ณ ์๋ค.
IntStream IntStream.range(int begin, int end)
IntStream IntStream.rangeClosed(int begin, int end)
๋์ ์ฐจ์ด์ ์ range()๋ ๋์ end๊ฐ ๋ฒ์์ ํฌํจ๋์ง ์๊ณ , rangeClosed()๋ ํฌํจ๋๋ค.
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()
์ฌ๊ธฐ์ ๋ฐํ๋๋ ์คํธ๋ฆผ๋ค์ ํฌ๊ธฐ๊ฐ ์ ํด์ง์ง์์ '๋ฌดํ ์คํธ๋ฆผ'์ด๋ฏ๋ก limit()๋ ๊ฐ์ด ์ฌ์ฉํด์ ์คํธ๋ฆผ์ ํฌ๊ธฐ๋ฅผ ์ ํํด์ฃผ์ด์ผ ์์ ํ ํ๋ก๊ทธ๋จ์ด ๋๋ค. limit()์ ์คํธ๋ฆผ์ ๊ฐ์๋ฅผ ์ง์ ํ๋๋ฐ ์ฌ์ฉ๋๋ฉฐ, ๋ฌดํ ์คํธ๋ฆผ์ ์ ํ ์คํธ๋ฆผ์ผ๋ก ๋ง๋ค์ด ์ค๋ค.
IntStream intStream = new Random().ints(); // ๋ฌดํ ์คํธ๋ฆผ
intStream.limit(5).forEach(System.out::println);
[์คํ๊ฒฐ๊ณผ]
1443532478
-2115253488
1079105334
941471356
178558647
๋ค๋ฅธ ๋ฐฉ๋ฒ์ผ๋ก๋ ๋งค๊ฐ๋ณ์๋ก ์คํธ๋ฆผ์ ํฌ๊ธฐ๋ฅผ ์ง์ ํ์ฌ ์ ํ ์คํธ๋ฆผ์ ๋ฐํํ๋๋ก ํ ์ ์๋ค.
IntStream ints(long streamSize)
LongStream longs(long streamSize)
DoubleStream doubles(long streamSize)
๋ํ ์์ ๋ฉ์๋๋ฅผ ํตํด ์์ฑ๋ ์คํธ๋ฆผ์ ๋์๋ ์๋์ ๋ฒ์๋ฅผ ๊ฐ๋๋ค.
Integer.MIN_VALUE <= ints() <= Integer.MAX_VALUE
Long.MIN_VALUE <= longs() <= Long.MAX_VALUE
0.0 <= doubles() < 1.0
์ง์ ๋ ๋ฒ์(begin ~ end)์ ๋์๋ฅผ ๋ฐ์์ํค๋ ์คํธ๋ฆผ์ ์ป๋ ๋ฉ์๋๋ ์๋์ ๊ฐ๋ค. ๋จ, 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)
๋๋ค์
Stream ํด๋์ค์ iterate()์ generate()๋ ๋๋ค์์ ๋งค๊ฐ๋ณ์๋ก ๋ฐ์์, ์ด ๋๋ค์์ ์ํด ๊ณ์ฐ๋๋ ๊ฐ๋ค์ ์์๋ก ํ๋ ๋ฌดํ ์คํธ๋ฆผ์ ์์ฑํ๋ค.
static Stream iterate(T seed, UnaryOperator f)
static Stream generate(Supplier s)
iterate()๋ ์จ์๊ฐ(seed)์ผ๋ก ์ง์ ๋ ๊ฐ๋ถํฐ ์์ํด์, ๋๋ค์ f์ ์ํด ๊ณ์ฐ๋ ๊ฒฐ๊ณผ๋ฅผ ๋ค์ seed๊ฐ์ผ๋ก ํด์ ๊ณ์ฐ์ ๋ฐ๋ณตํ๋ค. ์๋์ evenStream์ 0๋ถํฐ ์์ํด์ ๊ฐ์ด 2์ฉ ๊ณ์ ์ฆ๊ฐํ๋ค.
Stream evenStream = Stream.iterate(0, n->n+2); // 0, 2, 4, 6, ...
generate()๋ iterate()์ ๋ฌ๋ฆฌ ๊ณ์ฐ๋ ๊ฒฐ๊ณผ๋ฅผ ๋ค์ seed๊ฐ์ผ๋ก ํด์ ๊ณ์ฐํ์ง ์๋๋ค. ๋ํ ๋งค๊ฐ๋ณ์์ ํ์ ์ Supplier<T>๋ก ๋งค๊ฐ๋ณ์๊ฐ ์๋ ๋๋ค์๋ง์ ํ์ฉํ๋ค.
Stream randomStream = Stream.generate(Math::random);
Stream oneStream = Stream.generate(()->1));
โป ์ฃผ์์ฌํญ : iterate()์ generate()๋ก ์์ฑ๋ ์คํธ๋ฆผ์ ๊ธฐ๋ณธํ ์คํธ๋ฆผ ํ์
์ ์ฐธ์กฐ๋ณ์๋ก ๋ค๋ฃฐ ์ ์๋ค.
IntStream evenStream = Stream.iterate(0, n->n+2); // ์๋ฌ
DoubleStream randomStream = Stream.generate(Math::random); // ์๋ฌ
๊ตณ์ด ํ์ํ๋ค๋ฉด, ์๋์ ๊ฐ์ด mapToInt()์ ๊ฐ์ ๋ฉ์๋๋ก ๋ณํํด์ผ ํ๋ค. ๋ฐ๋๋ก IntStreamํ์ ์ ์คํธ๋ฆผ์ Stream<Integer>ํ์ ์ผ๋ก ๋ณํํ๋ ค๋ฉด, boxed()๋ฅผ ์ฌ์ฉํ๋ฉด ๋๋ค.
IntStream evenStream = Stream.iterate(0, n->n+2).mapToInt(Integer::valueOf);
Stream stream = evenStream.boxed(); // IntStream -> Stream
ํ์ผ
java.nio.file.Filesํจํค์ง์ ํ์ผ์ ๋ค๋ฃจ๋ ์ ์ฉํ ๋ฉ์๋๋ค์ด ์๋ค. ๊ทธ ์ค list()๋ ์ง์ ๋ ๋๋ ํ ๋ฆฌ(dir)์ ์๋ ํ์ผ์ ๋ชฉ๋ก์ ์์ค๋ก ํ๋ ์คํธ๋ฆผ์ ์์ฑํด์ ๋ฐํํ๋ค.
Stream Files.list(Path dir)
๊ทธ๋ฆฌ๊ณ , ํ์ผ์ ํ ํ์ ์์๋ก ํ๋ ์คํธ๋ฆผ์ ์์ฑํ๋ ๋ฉ์๋๋ ์๋ค.
Stream Files.lines(Path path)
Stream Files.lines(Path path, Charset cs)
Stream lines()
์๋ ์ธ๋ฒ์งธ ๋ฉ์๋๋ BufferdReaderํด๋์ค์ ์ํ ๊ฒ์ธ๋ฐ, ํ์ผ ๋ฟ๋ง ์๋๋ผ ๋ค๋ฅธ ์ ๋ ฅ๋์์ผ๋ก๋ถํฐ๋ ๋ฐ์ดํฐ๋ฅผ ํ๋จ์๋ก ์ฝ์ด์ฌ ์ ์๋ค.
๋น ์คํธ๋ฆผ
์์๊ฐ ํ๋๋ ์๋ ํ ๋น ์คํธ๋ฆผ์ ์์ฑํ ์๋ ์๋ค. ์คํธ๋ฆผ์ ์ฐ์ฐ์ ์ํํ ๊ฒฐ๊ณผ๊ฐ ํ๋๋ ์์ ๋, null๋ณด๋ค ๋น ์คํธ๋ฆผ์ ๋ฐํํ๋ ๊ฒ์ด ์๋ฌ๋ฅผ ์ค์ผ ์ ์๊ธฐ ๋๋ฌธ์ ๋ ๋ซ๋ค.
Stream emptyStream = Stream.empty(); // ๋น ์คํธ๋ฆผ ์์ฑ
long count = emptyStream.count() // 0
count()๋ ์์์ ๊ฐ์๋ฅผ ๋ฐํํ๋ค.
๋ ์คํธ๋ฆผ์ ์ฐ๊ฒฐ
Stream์ static๋ฉ์๋์ธ concat()์ ์ฌ์ฉํ๋ฉด, ๋ ์คํธ๋ฆผ์ ํ๋๋ก ์ฐ๊ฒฐํ ์ ์๋ค. ์ด ๋ ์ฐ๊ฒฐํ๋ ๋ ์คํธ๋ฆผ์ ์์๋ ๊ฐ์ ํ์ ์ด์ด์ผ ํ๋ค.
Stream strs1 = Stream.of(str1);
Stream strs2 = Stream.of(str2);
Stream strs3 = Stream.concat(stgr1, str2); // ๋ ์คํธ๋ฆผ์ ํ๋๋ก ์ฐ๊ฒฐ
3. ์คํธ๋ฆผ์ ์ค๊ฐ์ฐ์ฐ
์คํธ๋ฆผ ์๋ฅด๊ธฐ - skip(), limit()
์คํธ๋ฆผ ์ผ๋ถ๋ฅผ ์๋ฅผ ๋ skip()๊ณผ limit()์ ์ฌ์ฉํ๋ค. skip(3)์ ์ฒ์ 3๊ฐ์ ์์๋ฅผ ๊ฑด๋๋ฐ๊ณ , limit(5)๋ ์คํธ๋ฆผ์ ์์๋ฅผ ์ต๋ 5๊ฐ๋ก ์ ํํ๋ค.
Stream skip(long n)
Stream limit(long maxSize)
์๋ฅผ๋ค์ด 10๊ฐ์ ์์๋ฅผ ๊ฐ์ง ์คํธ๋ฆผ์ skip(3)๊ณผ limit(5)๋ฅผ ์ ์ฉํ๋ฉด 4๋ฒ์งธ ์์๋ถํฐ 5๊ฐ๋ฅผ ๊ฐ์ง ์คํธ๋ฆผ์ ๋ฐํํ๋ค.
IntStream intStream = IntStream.rangeClosed(1, 10); // 1~10์ ์์๋ฅผ ๊ฐ์ง ์คํธ๋ฆผ ์์ฑ
intStream.skip(3).limit(5).forEach(System.out::println);
[์คํ ๊ฒฐ๊ณผ]
4
5
6
7
8
๊ธฐ๋ณธํ ์คํธ๋ฆผ์๋ skip()๊ณผ limit()์ด ์ ์ ๋์ด ์๊ณ ๋ฐํ ํ์ ์ด ๊ธฐ๋ณธํ ์คํธ๋ฆผ์ด๋ผ๋ ๊ฒ๋ง ๋ค๋ฅด๋ค.
์คํธ๋ฆผ ์์ ๊ฑธ๋ฌ๋ด๊ธฐ - filter(), distinct()
distinct()๋ ์ค๋ณต์์๋ค์ ์ ๊ฑฐํ๊ณ filter()๋ ์กฐ๊ฑด์ ๋ง์ง ์๋ ์์๋ฅผ ๊ฑธ๋ฌ๋ธ๋ค. ์ฟผ๋ฆฌ์ ์์ฃผ ํก์ฌํ๋ค.
Stream filter(Predicate predicate)
filter๋ ๋งค๊ฐ๋ณ์๋ก Predicate๋ฅผ ํ์๋ก ํ๋๋ฐ, ์๋์ ๊ฐ์ด ์ฐ์ฐ๊ฒฐ๊ณผ๊ฐ boolean ์ธ ๋๋ค์์ ์ฌ์ฉํด๋ ๋๋ค.
IntStream intStream = IntStream.rangeClosed(1, 10); // 1~10์ ์์๋ฅผ ๊ฐ์ง ์คํธ๋ฆผ ์์ฑ
intStream.filter(i -> i%2 == 0).forEach(System.out::println); // 2 4 6 8 10
ํ์ํ๋ค๋ฉด filter()๋ฅผ ๋ค๋ฅธ ์กฐ๊ฑด์ผ๋ก ์ฌ๋ฌ ๋ฒ ์ฌ์ฉํ ์๋ ์๋ค.
์ ๋ ฌ - sorted()
์คํธ๋ฆผ์ ์ ๋ ฌํ ๋๋ sorted()๋ฅผ ์ฌ์ฉํ๋ฉด ๋๋ค.
Stream sorted()
Stream sorted(Comparator comparator)
sorted()๋ ์ง์ ๋ Comparator๋ก ์คํธ๋ฆผ์ ์ ๋ ฌํ๋๋ฐ, Comparator ๋์ int๊ฐ์ ๋ฐํํ๋ ๋๋ค์์ ์ฌ์ฉํด๋ ๋๋ค. Comparator๋ฅผ ์ง์ ํ์ง ์์ผ๋ฉด ์คํธ๋ฆผ ์์์ ๊ธฐ๋ณธ ์ ๋ ฌ ๊ธฐ์ค์ผ๋ก ์ ๋ ฌํ๋ค. ๋จ, ์คํธ๋ฆผ์ ์์๊ฐ Comparable์ ๊ตฌํํ ํด๋์ค๊ฐ ์๋๋ฉด ์์ธ๊ฐ ๋ฐ์ํ๋ค.
Stream strStream = Stream.of("dd", "aaa", "CC", "cc", "b");
strStream.sorted().forEach(System.out::println);
[์คํ๊ฒฐ๊ณผ]
CC
aaa
b
cc
dd
Comparator์ static๋ฉ์๋
naturalOrder()
reverseOrder()
comparing(Function(T, U) keyExtractor)
comparing(Function(T, U) keyExtractor, Comparator keyComparator)
comparingInt(ToIntFunction keyExtractor)
comparingLong(ToLongFunction keyExtractor)
comparingDouble(ToDoubleFunction keyExtractor)
nullsFirst(Comparator comparator)
nullsLast(Comparator comparator)
์ ๋ ฌ์ ์ฌ์ฉ๋๋ ๋ฉ์๋๋ ๋ง์ง๋ง ๊ฐ์ฅ ๊ธฐ๋ณธ์ ์ธ ๋ฉ์๋๋ comparing()์ด๋ค.
์คํธ๋ฆผ์ ์์๊ฐ Comparable์ ๊ตฌํํ ๊ฒฝ์ฐ, ๋งค๊ฐ๋ณ์ ํ๋์ง๋ฆฌ๋ฅผ ์ฌ์ฉํ๋ฉด ๋๊ณ ์๋๋ฉด ์ถ๊ฐ์ ์ธ ๋งค๊ฐ๋ณ์๋ก ์ ๋ ฌ๊ธฐ์ค(Comparator)๋ฅผ ๋ฐ๋ก ์ง์ ํด์ฃผ์ด์ผ ํ๋ค.
๋น๊ต ๋์์ด ๊ธฐ๋ณธํ์ธ ๊ฒฝ์ฐ, comparing()๋์ comparingInt, comparingLong, comparingDouble์ ์ฌ์ฉํ๋ฉด ์คํ ๋ฐ์ฑ๊ณผ ์ธ๋ฐ์ฑ ๊ณผ์ ์ด ์๊ธฐ ๋๋ฌธ์ ๋ ํจ์จ์ ์ด๋ค. ๊ทธ๋ฆฌ๊ณ ์ ๋ ฌ ์กฐ๊ฑด์ ์ถ๊ฐํ ๋์๋ thenComparing()์ ์ฌ์ฉํ๋ค.
thenComparing(Comparator other)
thenComparing(Function keyExtractor)
thenComparing(Function keyExtractor, Comparator keyComp)
๋ณํ - map()
์คํธ๋ฆผ์ ์์์ ์ ์ฅ๋ ๊ฐ ์ค์์ ์ํ๋ ํ๋๋ง ๋ฝ์๋ด๊ฑฐ๋ ํน์ ํํ๋ก ๋ณํํด์ผ ํ ๋๊ฐ ์๋๋ฐ ๊ทธ ๋ ์ฌ์ฉํ๋ ๊ฒ์ด ๋ฐ๋ก map()์ด๋ค. ์ด ๋ฉ์๋์ ์ ์ธ๋ถ๋ ์๋์ ๊ฐ๊ณ , Tํ์ ์ Rํ์ ์ผ๋ก ๋ณํํ๋ ํจ์๋ฅผ ์ง์ ํด์ผํ๋ค.
Stream map(Function mapper)
์๋ฅผ๋ค์ด File์คํธ๋ฆผ์์ ํ์ผ์ ์ด๋ฆ๋ง ๋ฝ์์ ์ถ๋ ฅํ๊ณ ์ถ์ ๋, ์๋์ ๊ฐ์ด map()์ ์ด์ฉํ๋ฉด ์ฝ๋ค.
Stream<File> fileStream = Stream.of(new File("Ex1.java"), new File("Ex1"),
new File("Ex1.bak"), new File("Ex2.java"), new File("Ex1.txt"));
// map()์ผ๋ก Stream์ Stream์ผ๋ก ๋ณํ
Stream filenameStream = fileStream.map(File::getName);
filenameStream.forEach(System.out::println); // ์คํธ๋ฆผ์ ๋ชจ๋ ํ์ผ์ ์ด๋ฆ์ ์ถ๋ ฅ
[์คํ๊ฒฐ๊ณผ]
Ex1.java
Ex1
Ex1.bak
Ex2.java
Ex1.txt
map()์ ์ค๊ฐ ์ฐ์ฐ์ผ๋ก ์คํธ๋ฆผ์ ๋ฐํํ๊ธฐ ๋๋ฌธ์ ํ๋์ ์คํธ๋ฆผ์ ์ฌ๋ฌ ๋ฒ ์ ์ฉํ ์๋ ์๊ณ , distinct๋ฅผ ๋ค์ ์ ์ฉํด์ ์ค๋ณต์ ์ ๊ฑฐํ ์๋ ์๋ค.
flatMap() - Stream<T[]>๋ฅผ Stream<T>๋ก ๋ณํ
์คํธ๋ฆผ์ ์์๊ฐ ๋ฐฐ์ด์ด๊ฑฐ๋ map()์ ์ฐ์ฐ๊ฒฐ๊ณผ๊ฐ ๋ฐฐ์ด์ธ ๊ฒฝ์ฐ, ์ฆ ์คํธ๋ฆผ์ ํ์ ์ด Stream<T[]>์ธ ๊ฒฝ์ฐ, Stream<T>๋ก ๋ค๋ฃจ๋ ๊ฒ์ด ๋ ํธ๋ฆฌํ ๋๊ฐ ์๋๋ฐ, ๊ทธ๋ด ๋๋ map()๋์ flatMap()์ ์ฌ์ฉํ๋ฉด ๋๋ค. ๋ค์ ์๋ฅผ ๋ณด์.
Stream strArrStrm = Stream.of(new String[] { "abc", "def", "ghi" }, new String[] { "ABc", "DEF", "GHI" });
์ด ๋ ๊ฐ ์์์ ๋ฌธ์์ด๋ค์ ํฉ์ณ์ ๋ฌธ์์ด์ ๋ค๋ฃจ๋ ์คํธ๋ฆผ, ์ฆ Stream<String>์ผ๋ก ๋ง๋ค๋ ค๋ฉด ์ด๋ป๊ฒ ํด์ผํ ๊น? for๋ฌธ์ ๋๋ ค์ ํ ์๋ ์๊ฒ ์ง๋ง flatMap์ ์ฌ์ฉํ๋ฉด ํ ์ค๋ก ๋๋ผ ์ ์๋ค.
String<String> strStrm = strArrStrm.flatMap(Arrays::stream);
๋ ๋ค๋ฅธ ์์ ๋ฅผ ๋ณด๋ฉด ์ฌ๋ฌ ๋ฌธ์ฅ์ ์์๋ก ํ๋ ์คํธ๋ฆผ์ด ์์ ๋, ์ด ๋ฌธ์ฅ๋ค์ split()์ผ๋ก ๋๋ ์ ์์๊ฐ ๋จ์ด์ธ ์คํธ๋ฆผ์ ๋ง๋ค๊ณ ์ถ์ ๊ฒฝ์ฐ๊ฐ ์์ ๊ฒ์ด๋ค.
String[] lineArr = {
"Belive or not It is true",
"Do or do not There is no try"
};
Stream<String> lineStream = Arrays.stream(lineArr);
// Stream strArrStream = lineStream.map(1->Stream.of(line.split(" +")));
Stream<String[]> strArrStream = lineStream.flatMap(1->Stream.of(line.split(" +")));
์ด๋ด ๋ map์ Stream<String[]>์ ๋๋ ค์ฃผ๊ธฐ ๋๋ฌธ์ ์ ์ ํ์ง ์๊ณ flatMap์ผ๋ก ํ๊ฒ๋๋ฉด ์ํ๋ Stream<String>์ ๋ฝ์๋ผ ์ ์๋ค.
๋๋ฌผ์ง๋ง, ์คํธ๋ฆผ์ ์์๋ก ํ๋ ์คํธ๋ฆผ, ์ฆ ์คํธ๋ฆผ์ ์คํธ๋ฆผ์ ํ๋์ ์คํธ๋ฆผ์ผ๋ก ํฉ์น ๋๋ flatMap()์ ์ฌ์ฉํ๋ค.
Stream<String> strStrm = Stream.of("abc", "def", "jkl");
Stream<String> strStrm2 = Stream.of("ABC", "DEF", "JKL");
Stream<Stream<String>> strmStrm = Stream.of(strStrm, strStrm2);
์ด ๋ Stream<Stream<String>>์ Stream<String>์ผ๋ก ๋ณํํ๋ ค๋ฉด ๋ค์๊ณผ ๊ฐ์ด map()๊ณผ flatMap()์ ํจ๊ป ์ฐ๋ฉด ๋๋ค.
Stream<String> strStream = strmStrm
.map(s -> s.toArray(String[]::new)) // Stream<Stream<String>> -> Stream<String[]>
.flatMap(Arrays::stream); // Stream<String[]> -> Stream<String>
toArray()๋ ์คํธ๋ฆผ์ ๋ฐฐ์ด๋ก ๋ณํํด์ ๋ฐํํ๋๋ฐ ๋งค๊ฐ๋ณ์๋ฅผ ์ง์ ํ์ง ์์ผ๋ฉด Object[]์ ๋ฐํํ๊ธฐ ๋๋ฌธ์ ์์ ๊ฐ์ด ํน์ ํ์ ์ ์ฌ์ฉํ๋ ค๋ฉด ํด๋น ํ์ ์ ์์ฑ์๋ฅผ ์ง์ ํด์ฃผ์ด์ผํ๋ค.
4. ์ต์ข ์ฐ์ฐ
๋ฆฌ๋์ฑ - reduce()
์คํธ๋ฆผ์ ์์๋ฅผ ์ค์ฌ๋๊ฐ๋ฉด์ ์ฐ์ฐ์ ์ํํ๊ณ ์ต์ข ๊ฒฐ๊ณผ๋ฅผ ๋ฐํํ๋ค. ๊ทธ๋์ ๋งค๊ฐ๋ณ์์ ํ์ ์ด BinaryOperator<T>์ด๋ค. ์ฒ์ ๋ ์์๋ฅผ ๊ฐ์ง๊ณ ์ฐ์ฐํ๊ณ ๊ทธ ๊ฒฐ๊ณผ๋ฅผ ๋ค์์์์ ์ฐ์ฐํ๋ค. ์ด ๊ณผ์ ์์ ํ๋์ฉ ์์๋ฅผ ์๋ชจํ๊ณ ์คํธ๋ฆผ์ ๋ชจ๋ ์์๋ฅผ ์๋ชจํ ํ์ ๊ทธ ๊ฒฐ๊ณผ๋ฅผ ๋ฐํํ๋ค.
Optional<T> reduce(BinaryOperator<T> accumulator)
์ฌ๊ธฐ์ Optional<T>์ด๋ ์ต์ข ์ฐ์ฐ์ ๊ฒฐ๊ณผ ํ์ ์ด ์์ ๊ฐ์ด Optional์ธ ๊ฒฝ์ฐ๊ฐ ์๋๋ฐ ์ด๋ ์ง๋ค๋ฆญ ํด๋์ค๋ก "Tํ์ ์ ๊ฐ์ฒด"๋ฅผ ๊ฐ์ธ๋ ๋ํผ ํด๋์ค์ด๋ค. ๊ทธ๋์ Optionalํ์ ์ ๊ฐ์ฒด์๋ ๋ชจ๋ ํ์ ์ ์ฐธ์กฐ๋ณ์๋ฅผ ๋ด์ ์ ์๋ค. (JDK1.8๋ถํฐ ์ถ๊ฐ๋จ)
public final class Optional<T> {
private final T value; // T ํ์
์ ์ฐธ์กฐ๋ณ์
...
}
์ต์ข ์ฐ์ฐ์ ๊ฒฐ๊ณผ๋ฅผ ๊ทธ๋ฅ ๋ฐํํ๋ ๊ฒ์ด ์๋๋ผ Optional๊ฐ์ฒด์ ๋ด์์ ๋ฐํํ๋ฏ๋ก ๋ฐํ๋ ๊ฒฐ๊ณผ๊ฐ null์ธ์ง ๋งค๋ฒ if๋ฌธ์ผ๋ก ์ฒดํฌํ๋ ๋์ Optional์ ์ ์๋ ๋ฉ์๋๋ฅผ ํ์ฉํด์ ๊ฐ๋จํ๊ฒ ์ฒ๋ฆฌํ ์ ์๋ค. ์ด๋ก ์ธํด ๋ ์ฒดํฌ๋ฅผ ์ํ if๋ฌธ ์์ด๋ NullPointerException์ด ๋ฐ์ํ์ง ์๋ ๋ณด๋ค ๊ฐ๊ฒฐํ๊ณ ์์ ํ ์ฝ๋๋ฅผ ์์ฑํ๋ ๊ฒ์ด ๊ฐ๋ฅํด์ง๋ค.
โป ์ฐธ๊ณ : Objectsํด๋์ค์ isNull(), nonNull(), requireNonNull()๊ณผ ๊ฐ์ ๋ฉ์๋๊ฐ ์๋ ๊ฒ๋ ๋ ์ฒดํฌ๋ฅผ ์ํ if๋ฌธ์ ๋ฉ์๋ ์์ ๋ฃ์ด์ ์ฝ๋์ ๋ณต์ก๋๋ฅผ ๋ฎ์ถ๊ธฐ ์ํ ๊ฒ.
๋ค์ ๋์์์ ์ฐ์ฐ๊ฒฐ๊ณผ์ ์ด๊ธฐ๊ฐ์ ๊ฐ๋ reduce()๋ ์๋๋ฐ, ์ด ๋ฉ์๋๋ค์ ์ด๊ธฐ๊ฐ๊ณผ ์คํธ๋ฆผ์ ์ฒซ ๋ฒ์งธ ์์๋ก ์ฐ์ฐ์ ์์ํ๋ค. ์์๊ฐ ํ๋๋ ์๋ ๊ฒฝ์ฐ ์ด๊ธฐ๊ฐ์ด ๋ฐํ๋๊ธฐ ๋๋ฌธ์ ๋ฐํ ํ์ ์ด Optional<T>๊ฐ ์๋๋ผ T์ด๋ค.
T reduce(T identity, Binaryoperator<T> accumulator)
U reduce(U identity, BiFunction<U,T,U> accumulator, Binaryoperator<U> combiner)
์์ ๋๋ฒ์งธ ๋ฉ์๋์ ๋ง์ง๋ง ๋งค๊ฐ๋ณ์์ธ combiner๋ ๋ณ๋ ฌ ์คํธ๋ฆผ์ ์ํด ์ฒ๋ฆฌ๋ ๊ฒฐ๊ณผ๋ฅผ ํฉ์น ๋์ฌ์ฉํ๊ธฐ ์ํ ๊ฒ์ด๋ค.
int count = intStream.reduce(0, (a,b) -> a+1); // count()
int sum = intStream.reduce(0, (a,b) -> a+b); // sum()
int max = intStream.reduce(0, (a,b) -> a>b ? a:b); // max()
int min = intStream.reduce(Integer.MIN_VALUE, (a,b) -> a
min, max๋ ์ด๊ธฐ๊ฐ์ด ํ์ ์๊ธฐ ๋๋ฌธ์ ์์์ ์ธ๊ธํ Optional<T>๋ฅผ ๋ฐํํ๋ ๋ฉ์๋๋ฅผ ์ฌ์ฉํ๋ ๊ฒ์ด ๋ ๋ซ๋ค. ๋จ intStream์ ํ์ ์ด IntStream์ธ ๊ฒฝ์ฐ OptionalInt๋ฅผ ๋ฐํํ๋ค.
T reduce(T identity, BinaryOperator<T> accumulator) {
T a = identity;
for (T b : stream)
a = accumulator.apply(a, b);
return a;
}
5. collect()
์คํธ๋ฆผ์ ์ต์ข ์ฐ์ฐ ์ค ๊ฐ์ฅ ๋ณต์กํ๋ฉด์ ์ ์ฉํ๊ฒ ์ฌ์ฉํ ์ ์๋ ๊ฒ์ด collect()์ด๋ค.
์คํธ๋ฆผ์ ์์๋ฅผ ์์งํ๊ธฐ ์ํ ๋ฉ์๋๋ก ์์ง ๋ฐฉ๋ฒ์ ๋ํด ์ ์๋์ด ์์ด์ผ ํ๋๋ฐ ์ด ๋ฐฉ๋ฒ์ ์ ์ํ ๊ฒ์ด ๋ฐ๋ก ์ปฌ๋ ํฐ(collector)์ด๋ค.
์ปฌ๋ ํฐ๋ Collector์ธํฐํ์ด์ค๋ฅผ ๊ตฌํํ ๊ฒ์ผ๋ก, ์ง์ ๊ตฌํํ ์๋ ์๊ณ ๋ฏธ๋ฆฌ ์์ฑ๋ ๊ฒ์ ์ฌ์ฉํ ์๋ ์๋ค.
collect() ์คํธ๋ฆผ์ ์ต์ข ์ฐ์ฐ, ๋งค๊ฐ๋ณ์๋ก ์ปฌ๋ ํฐ๋ฅผ ํ์๋ก ํ๋ค.
Collector ์ธํฐํ์ด์ค, ์ปฌ๋ ํฐ๋ ์ด ์ธํฐํ์ด์ค๋ฅผ ๊ตฌํํด์ผ ํ๋ค.
Collectors ํด๋์ค, static๋ฉ์๋๋ก ๋ฏธ๋ฆฌ ์์ฑ๋ ์ปฌ๋ ํฐ๋ฅผ ์ ๊ณตํ๋ค.
collect()๋ ๋งค๊ฐ๋ณ์์ ํ์ ์ด Collector์ธ๋ฐ ๋งค๊ฐ๋ณ์๊ฐ Collector๋ฅผ ๊ตฌํํ ํด๋์ค์ ๊ฐ์ฒด์ด์ด์ผ ํ๋ค๋ ๋ป์ด๋ค. ๋งค๊ฐ๋ณ์๋ก ๋์ด์จ ๊ฐ์ฒด์ ๊ตฌํ๋ ๋ฐฉ๋ฒ๋๋ก ์คํธ๋ฆผ์ ์์๋ฅผ ์์งํ๋ค.
Object collect(Collector collector)
Object collect(Supplier supplier, BiConsumer accumulator, BiConsumer combiner)
์๋์ collect()๋ ์ ์ฌ์ฉ๋์ง ์์ง๋ง , Collector์ธํฐํ์ด์ค๋ฅผ ๊ตฌํํ์ง ์๊ณ ๊ฐ๋จํ ๋๋ค์์ผ๋ก ์์งํ ๋ ์ฌ์ฉํ๋ฉด ํธ๋ฆฌํ๋ค๊ณ ํ๋ค.
์คํธ๋ฆผ์ ์ปฌ๋ ์ ๊ณผ ๋ฐฐ์ด๋ก ๋ณํ - toList(), toSet(), toCollection(), toArray()
์คํธ๋ฆผ์ ๋ชจ๋ ์์๋ฅผ ์ปฌ๋ ์ ์ ์์งํ๋ ค๋ฉด, Collectorsํด๋์ค์ toList()์ ๊ฐ์ ๋ฉ์๋๋ฅผ ์ฌ์ฉํ๋ฉด ๋๋ค. List๋ Set์ด ์๋ ํน์ ์ปฌ๋ ์ ์ ์ง์ ํ๋ ค๋ฉด, toCollection()์ ํด๋น ์ปฌ๋ ์ ์ ์์ฑ์ ์ฐธ์กฐ๋ฅผ ๋งค๊ฐ๋ณ์๋ก ๋ฃ์ด์ฃผ๋ฉด ๋๋ค.
List<String> names = stuStream.map(Student::getName).collect(Collectors.toList());
ArrayList<String> list = names.stream().collect(Collectors.toCollection(ArrayList::new));
Map์ ์ฌ์ฉํ๋ ๊ฒฝ์ฐ์๋ ํค์ ๊ฐ์ ์์ผ๋ก ์ ์ฅํด์ผ ํ๋ฏ๋ก ๊ฐ์ฒด์ ์ด๋ค ํ๋๋ฅผ ํค๋ก ์ฌ์ฉํ ์ง, ๊ฐ์ผ๋ก ์ฌ์ฉํ ์ง๋ฅผ ์ง์ ํด์ฃผ์ด์ผํ๋ค.
๋ฆฌ๋์ฑ, ํต๊ณ - counting(), summmingInt(), averagingInt(), maxBy(), minBy(), reducing()
์ต์ข ์ฐ์ฐ์ ์ด๋ฏธ ์กด์ฌํ๋ ์ฐ์ฐ๋ค์ collect()๋ก ๊ตฌํํ ์ ์๋ค.
// int sum = intStream.reduce(0, (a,b) -> a+b);
int sum = intStream.boxed().collect(reducing(0, (a,b) -> a+b));
boxed()์ ์ฌ์ฉํ ์ด์ ๋ IntStream์๋ ๋งค๊ฐ๋ณ์๊ฐ 3๊ฐ์ง๋ฆฌ collect()๋ง ์ ์๋์ด ์๊ธฐ ๋๋ฌธ์ ๋ฐ์ฑ์ ํตํด Stream<Integer>๋ก ๋ณํํด์ฃผ์ด์ผ ๋งค๊ฐ๋ณ์ 1๊ฐ์ง๋ฆฌ collect()๋ฅผ ์ฌ์ฉํ ์ ์๊ธฐ ๋๋ฌธ์ด๋ค.
๋ฌธ์์ด ๊ฒฐํฉ - joining()
๋ฌธ์์ด ์คํธ๋ฆผ์ ๋ชจ๋ ์์๋ฅผ ํ๋์ ๋ฌธ์์ด๋ก ์ฐ๊ฒฐํด์ ๋ฐํํ๋ค. ๊ตฌ๋ถ์๋ฅผ ์ง์ ํด์ค ์๋ ์๊ณ , ์ ๋์ฌ์ ์ ๋ฏธ์ฌ๋ ์ง์ ๊ฐ๋ฅํ๋ค. ๋์ ์คํธ๋ฆผ์ ์์๊ฐ String์ด๋ StringBuffer์ฒ๋ผ ChareSequence์ ์์์ธ ๊ฒฝ์ฐ๋ง ๊ฐ๋ฅํ๋ค. ๋ฐ๋ผ์ ์คํธ๋ฆผ์ ์์๊ฐ ๋ฌธ์์ด์ด ์๋ ๊ฒฝ์ฐ 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()์์ด ์คํธ๋ฆผ์ ๋ฐ๋ก ์ฌ์ฉํ๊ฒ ๋๋ฉด ์คํธ๋ฆผ์ ์์์ toString()์ ํธ์ถํ ๊ฒฐ๊ณผ๋ฅผ ์ฌ์ฉํ๋ค.
๊ทธ๋ฃนํ์ ๋ถํ - groupingBy(), partitioningBy()
์ฌ๊ธฐ๋ถํฐ collect()์ ์ ์ฉํจ์ ์ ์ ์๋ ๋ถ๋ถ์ด๋ค. ๊ทธ๋ฃนํ๋ ์คํธ๋ฆผ์ ์์๋ฅผ ํน์ ๊ธฐ์ค์ผ๋ก ๊ทธ๋ฃนํํ๋ ๊ฒ์ ์๋ฏธํ๊ณ ๋ถํ ์ ์คํธ๋ฆผ์ ์์๋ฅผ ๋ ๊ฐ์ง ์ง์ ๋ ์กฐ๊ฑด์ ์ผ์นํ๋ ๊ทธ๋ฃน๊ณผ ์ผ์นํ์ง ์๋ ๊ทธ๋ฃน์ผ๋ก์ ๋ถํ ์ ์๋ฏธํ๋ค.
groupingBy()๋ ์คํธ๋ฆผ์ ์์๋ฅผ Function์ผ๋ก, partitioningBy()๋ Predicate๋ก ๋ถ๋ฅํ๋ค.
Collector groupingBy(Function classifier)
Collector groupingBy(Function classifier, Collector downstream)
Collector groupingBy(Function classifier, Supplier mapFactory, Collector downstream)
Collector partitioningBy(Predicate predicate)
Collector partitioningBy(Predicate predicate, Collector downstream)
๋ฉ์๋ ์ ์๋ฅผ ๋ดค์ ๋ ๋งค๊ฐ๋ณ์์ ์ฐจ์ด ๋ง๊ณ ๋ ๋์ผํ๋ค๋ ๊ฒ์ ์ ์ ์๋ค. ์คํธ๋ฆผ์ ๋ ๊ฐ์ ๊ทธ๋ฃน์ผ๋ก ๋๋ ์ผ ํ๋ค๋ฉด ๋น์ฐํ partitioningBy()๋ก ๋ถํ ํ๋ ๊ฒ์ด ๋ ๋น ๋ฅด๋ค. ๊ทธ ์ธ์๋ groupingBy()๋ฅผ ์ฌ์ฉํ๋ฉด ๋๋ค. ๋๊ฐ ๋ค ๊ฒฐ๊ณผ๋ Map์ ๋ด๊ฒจ ๋ฐํ๋๋ค.
์์์ ์ฌ์ฉ๋ Studentํด๋์ค
class Student {
String name; // ์ด๋ฆ
boolean isMale; // ์ฑ๋ณ
int hak; // ํ๋
int ban; // ๋ฐ
int score; // ์ ์
Student(String name, boolean isMale, int hak, int ban, int score) {
this.name = name;
this.isMale = isMale;
this.hak = hak;
this.ban = ban;
this.score = score;
}
String getName() { return name;}
boolean isMale() { return isMale;}
int getHak() { return hak;}
int getBan() { return ban;}
int getScore() { return score;}
public String toString() {
return String.format("[%s, %s, %dํ๋
%d๋ฐ, %3d์ ]", name, isMale ? "๋จ":"์ฌ", hak, ban, score);
}
enum Level { // ์ฑ์ ์ ์, ์ค, ํ 3๋จ๊ณ๋ก ๋ถ๋ฅ
HIGH, MID, LOW
}
} // class Student
๊ทธ๋ฆฌ๊ณ ์คํธ๋ฆผ stuStudent๋ ์๋์ ๊ฐ์ ์์๋ค์ค ์ด๋ฃจ์ด์ ธ ์๋ค๊ณ ํ์.
Stream<Student> stuStream = Stream.of(
new Student("๋์๋ฐ", true, 1, 1, 300),
new Student("๊น์ง๋ฏธ", false, 1, 1, 250),
new Student("๊น์๋ฐ", true, 1, 1, 200),
new Student("์ด์ง๋ฏธ", false, 1, 2, 150),
new Student("๋จ์๋ฐ", true, 1, 2, 100),
new Student("์์ง๋ฏธ", false, 1, 2, 50),
new Student("ํฉ์ง๋ฏธ", false, 1, 3, 100),
new Student("๊ฐ์ง๋ฏธ", false, 1, 3, 150),
new Student("์ด์๋ฐ", true, 1, 3, 200),
new Student("๋์๋ฐ", true, 2, 1, 300),
new Student("๊น์ง๋ฏธ", false, 2, 1, 250),
new Student("๊น์๋ฐ", true, 2, 1, 200),
new Student("์ด์ง๋ฏธ", false, 2, 2, 150),
new Student("๋จ์๋ฐ", true, 2, 2, 100),
new Student("์์ง๋ฏธ", false, 2, 2, 50),
new Student("ํฉ์ง๋ฏธ", false, 2, 3, 100),
new Student("๊ฐ์ง๋ฏธ", false, 2, 3, 150),
new Student("์ด์๋ฐ", true, 2, 3, 200)
);
partitioningBy()์ ์ํ ๋ถ๋ฅ
๊ฐ์ฅ ๊ธฐ๋ณธ์ ์ธ ๋ถํ ์ ํ์์ ๋ณด๊ฐ ์๋ค๊ณ ํ์ ๋ ํ์๋ค์ ์ฑ๋ณ๋ก ๋๋๋ ๊ฒ์ด๋ค.
Map<Boolean, List<Student>> stuBySex = stuStream.collect(partitioningBy(Student::isMale));
List<Student> maleStudent = stuBySex.get(true);
List<Student> femaleStudent = stuBySex.get(false);
counting()์ ์ถ๊ฐํด์ ๊ฐ๊ฐ ํ์์ ์๋ฅผ ๊ตฌํ ์๋ ์๋ค.
Map<Boolean, List<Student>> stuNumBySex = stuStream
.collect(partitioningBy(Student::isMale), counting());
System.out.println("๋จํ์ ์ : " + stuNumBySex.get(true)); // 8
System.out.println("์ฌํ์ ์ : " + stuNumBySex.get(false)); // 10
groupingBy()์ ์ํ ๋ถ๋ฅ
partitioningBy()์ ์ฌ์ฉ๋ฒ์ ๊ฐ๋ค. ๋ฐ ๋ณ๋ก ๊ทธ๋ฃน์ง์ด Map์ ์ ์ฅํ๋ ๋ฐฉ๋ฒ์ ๋ณด์.
Map<Boolean, List<Student>> stuByBan = stuStream
.collect(groupingBy(Student::getBan));
groupingBy()๋ก ๊ทธ๋ฃนํ๋ฅผ ํ๋ฉด ๊ธฐ๋ณธ์ ์ผ๋ก List<T>์ ๋ด๋๋ค. ๊ทธ๋์ ์์ ๋ฌธ์ฅ์ toList()๊ฐ ์๋ต๋ ํํ๋ค.
Map<Boolean, List<Student>> stuByBan = stuStream
.collect(groupingBy(Student::getBan), toList());
toList() ๋์ toSet()์ด๋ toCollection(HashSet::new) ์ฌ์ฉ ๊ฐ๋ฅํ๋ค. ๋จ Map์ ์ง๋ค๋ฆญํ์ ๋ ๊ทธ์ ๋ง์ถฐ ๋ฐ๊พธ์ด์ผ ํ๋ค.
์ฐธ์กฐ : ์๋ฐ์ ์ ์ 3rd