pluck
주어진 key 값에 해당하는 값으로 이루어진 리스트를 리턴한다.
/*요소에서 특정 값만 추출하는 함수 */
function pluck(list, key){
return map(list, obj => obj[key]);
}
위와 같이 동적 언어인 자바스크립트 처럼 유연하게 대응하기가 자바는 힘들다.
먼저 타입 T 는 호출할 때마다 달라지고, 호출자에 의해 타입이 결정된다. 그래서 런타임 시 클래스 정보를 얻어올 수 있는 리플랙션 API를 사용해 값을 얻어왔다. 이때 리턴 타입은 Object 다. 그래서 추가로 타입 토큰을 인자로 받아 형변환에 사용했다.
static <T,R> List<R> pluck(List<T> list, String key,Class<R> typeToken){
List<R> result = map(list, val-> pluckHelper(val, key, typeToken));
return some(result, Objects::isNull).booleanValue() ? Collections.emptyList() :result;
}
//자바스크립트 같이 동적 언어와 다르게 자바는 정적 언어로 아래와 같이 타입토큰으로 형변환을 했다.
static <T,R> R pluckHelper(T val,String key,Class<R> typeToken) {
try {
Field field = val.getClass().getDeclaredField(key);
return typeToken.cast(field.get(val));
} catch (Exception e) {
return null;
}
}
/*Stream 클래스 메서드*/
<R> List<R> pluck(String key,Class<R> typeToken){
Stream<R> result = map(val-> pluckHelper(val, key, typeToken));
return result.some(Objects::isNull).booleanValue() ? Collections.emptyList() :result.toList();
}
<R> R pluckHelper(T val,String key,Class<R> typeToken) {
try {
Field field = val.getClass().getDeclaredField(key);
return typeToken.cast(field.get(val));
} catch (Exception e) {
return null;
}
}
System.out.println("pluck 테스트");
System.out.println(pluck(users, "age",Integer.class));
//타입불일치
System.out.println(pluck(users, "age",String.class));
System.out.println(Stream.stream(users).pluck("age",Integer.class));
//타입불일치
System.out.println(Stream.stream(users).pluck("age",String.class));
pluck 테스트
[36, 32, 32, 27, 25, 26, 31, 23]
[]
[36, 32, 32, 27, 25, 26, 31, 23]
[]
reject
/*
* filter 정반대, 주어진 조건에 해당하지 않은 값만 걸러낸다.
*/
static <T> List<T> reject(List<T> list, Predicate<T> predi){
return filter(list, predi.negate());
}
/*Stream 클래스 메서드*/
Stream<T> reject(Predicate<T> predi){
return filter(predi.negate());
}
전체 코드
package javabasic.stream;
import java.lang.reflect.Field;
import java.util.*;
import java.util.function.*;
import javabasic.stream.FuncEx01.Stream;
public class FuncEx01 {
static class User{
final Long id;
final String name;
final Integer age;
public User(Long id, String name, Integer age) {
this.id = id;
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "[id=" + id + ", name=" + name + ", age=" + age + "]\n";
}
}
static List<User> getUsers(){
return Arrays.asList(
new User(10L, "ID", 36),
new User(20L, "BJ", 32),
new User(30L, "JM", 32),
new User(40L, "PJ", 27),
new User(50L, "HA", 25),
new User(60L, "JE", 26),
new User(70L, "JI", 31),
new User(80L, "MP", 23)
);
}
public static void main(String[] args) {
List<User> users = getUsers();
/*
System.out.println("filter = "+
filter(users,user->user.age>30)+"\n"
+users);
System.out.println("map = "+
map(users, user->user.id));
each(Arrays.asList(1,2,3,4,5),System.out::print);
System.out.println();
System.out.println("reduce = "+
reduce(Arrays.asList(1,2,3,4,5), (a,b)->a+b, Integer.valueOf(0)));
System.out.println("reduce = "+
reduce(Arrays.asList(1,2,3,4,5), (a,b)->a+b));
System.out.println("reduce = "+
reduce(Arrays.asList(1), (a,b)->a+b, Integer.valueOf(0)));
System.out.println("reduce = "+
reduce(Arrays.asList(1), (a,b)->a+b));
List<String> result = Stream.stream(users)
.filter(user->user.age > 30)
.map(user->user.name)
.toList();
System.out.println(result);
//메서드 체이닝 방식과 차이 비교
System.out.println("default = "+
reduce(
map(
filter(users,user->user.age>30)
, user->user.age)
, (a,b)->a+b));
//기존은 안쪽에서부터 밖으로 나온다. 중첩구조가 심해질 수록 가독성이 안좋아진다.
//스트림을 통한 메서드 체이닝 방식은 순차적으로 적용된다.
System.out.println("stream = " +
Stream.stream(users)
.filter(user->user.age > 30)
.map(user->user.age)
.reduce(Integer::sum)
// .reduce((a,b)->a+b)
);
*/
//find, findIndex 실습
System.out.println(find(users, user->user.age<32));
System.out.println(find(users, user->user.age>55));
System.out.println(findIndex(users, user->user.age<32));
System.out.println(findIndex(users, user->user.age>55));
System.out.println(Stream.stream(users).find(user->user.age<32));
System.out.println(Stream.stream(users).find(user->user.age>55));
System.out.println(Stream.stream(users).findIndex( user->user.age<32));
System.out.println(Stream.stream(users).findIndex( user->user.age>55));
System.out.println("every , find 테스트");
System.out.println(some(users, u->u.age>30));
System.out.println(some(users, u->u.age>50));
System.out.println(every(users, u->u.age>30));
System.out.println(every(users, u->u.age>10));
System.out.println(Stream.stream(users).some(u->u.age>30));
System.out.println(Stream.stream(users).some(u->u.age>50));
System.out.println(Stream.stream(users).every(u->u.age>30));
System.out.println(Stream.stream(users).every(u->u.age>10));
System.out.println("pluck 테스트");
System.out.println(pluck(users, "age",Integer.class));
//타입불일치
System.out.println(pluck(users, "age",String.class));
System.out.println(Stream.stream(users).pluck("age",Integer.class));
//타입불일치
System.out.println(Stream.stream(users).pluck("age",String.class));
System.out.println("reject 테스트");
System.out.println(reject(users,u->u.age>30));
System.out.println(Stream.stream(users).reject(u->u.age>30).toList());
}
/*
* filter는 처리 결과가 입력 결과와 타입은 같다.
* 길이는 같거나, 작을 수 밖에 없다.
*/
static <T> List<T> filter(List<T> list, Predicate<T> predi) {
//함수형 프로그래밍은 원본 데이터를 수정하지 않는다. 새로운 데이터를 리턴하여
//부수효과를 극단적으로 배제한다.
ArrayList<T> newList = new ArrayList<>();
each(list,data->{
if(predi.test(data))
newList.add(data);
});
return newList;
}
/*
* filter 정반대, 주어진 조건에 해당하지 않은 값만 걸러낸다.
*/
static <T> List<T> reject(List<T> list, Predicate<T> predi){
return filter(list, predi.negate());
}
/*
* map은 처리 결과가 입력 결과와 타입은 같거나 다르다.
* 길이는 항상 같다.
*/
static <T,R> List<R> map(List<T> list, Function<T, R> mapper){
ArrayList<R> newList = new ArrayList<>();
each(list, data->newList.add(mapper.apply(data)));
return newList;
}
/*
* 반복문을 중복을 대체할 함수
*/
static <T> List<T> each(List<T> list, Consumer<T> iter){
for(T data : list)
iter.accept(data);
return list;
}
/*
* 자바는 스트림 안에서 외부 변수를 참조하면 그 변수는 final 속성이 된다.
* 따라서 갱신하면서 누산을 하지 못해, 간접적으로 값을 수정해야 한다.
*/
static <T> T reduce(List<T> list, BinaryOperator<T> reducer ,T memo) {
//간소화된 유효성 검사, 본질을 흐리지 않는 선에서 간략화
if(memo == null ) return reduce(list.subList(1, list.size()), reducer, list.get(0));
if(list.size() < 2) return list.get(0);
HashMap<Class<?>, T> map = new HashMap<>();
map.put(memo.getClass(), memo);
each(list, data->map.compute(memo.getClass(), (k,v)->reducer.apply(v, data)));
return map.get(memo.getClass());
}
/*
* 자바는 람다에서 사용하는 변수는 final 속성을 띄기 때문에 memo 값 처리가 애매해진다.
* 삼항연산자 등 별도 처리하기보다 오버로딩으로 처리했다.
*/
static <T> T reduce(List<T> list, BinaryOperator<T> reducer) {
if(list.size() < 2) return list.get(0);
return reduce(list.subList(1, list.size()), reducer, list.get(0));
}
/*
* null 값을 간접적으로 다루기 위한 래퍼클래스 사용
*/
static <T> Optional<T> find(List<T> list, Predicate<T> predi) {
for(int i=0;i<list.size();i++) {
T value = list.get(i);
if(predi.test(value)) return Optional.of(value);
}
return Optional.empty();//편의상 null리턴
}
static <T> Integer findIndex(List<T> list, Predicate<T> predi) {
for(int i=0;i<list.size();i++) {
if(predi.test(list.get(i))) return i;
}
return -1;
}
static <T> Boolean some(List<T> list, Predicate<T> predi) {
return findIndex(list, predi) != -1;
}
static <T> Boolean every(List<T> list, Predicate<T> predi) {
return findIndex(list, predi.negate()) == -1;
// return findIndex(list, val->!predi.test(val)) == -1;
}
static <T> Boolean contains(List<T> list, T data){
return findIndex(list, val-> Objects.equals(val, data) ) != -1 ;
}
static <T,R> List<R> pluck(List<T> list, String key,Class<R> typeToken){
List<R> result = map(list, val-> pluckHelper(val, key, typeToken));
return some(result, Objects::isNull).booleanValue() ? Collections.emptyList() :result;
}
//런타임 시 타입이 확정되는 경우 어쩔 수 없이 Object를 리턴해야 한다.
static <T,R> R pluckHelper(T val,String key,Class<R> typeToken) {
try {
Field field = val.getClass().getDeclaredField(key);
return typeToken.cast(field.get(val));
} catch (Exception e) {
return null;
}
}
/*
* pipe, go 구현을 시도하려 했지만, 근본적으로 자바는 함수가 개념이 없어
* 호출부를 단일로 추상화할 수 없다. apply, test ...
* 따라서 하나의 클래스로 묶었다.
*/
static class Stream<T> {
List<T> list;
static <T> Stream<T> stream(List<T> list){
Stream<T> stream = new Stream<>();
stream.list = list;
return stream;
}
Stream<T> filter(Predicate<T> predi) {
//함수형 프로그래밍은 원본 데이터를 수정하지 않는다. 새로운 데이터를 리턴하여
//부수효과를 극단적으로 배제한다.
ArrayList<T> newList = new ArrayList<>();
forEach(data->{
if(predi.test(data))
newList.add(data);
});
return stream(newList);
}
Stream<T> reject(Predicate<T> predi){
return filter(predi.negate());
}
<R> Stream<R> map(Function<T, R> mapper){
ArrayList<R> newList = new ArrayList<>();
forEach(data->newList.add(mapper.apply(data)));
return stream(newList);
}
void forEach(Consumer<T> iter){
for(T data : list) iter.accept(data);
}
T reduce(BinaryOperator<T> reducer ,T memo) {
//간소화된 유효성 검사, 본질을 흐리지 않는 선에서 간략화
if(memo == null) return reduce(reducer);
if(this.list.size() < 2) return this.list.get(0);
HashMap<Class<?>, T> map = new HashMap<>();
map.put(memo.getClass(), memo);
each(this.list, data->map.compute(memo.getClass(), (k,v)->{
T result = reducer.apply(v, data);
return result;
}));
return map.get(memo.getClass());
}
T reduce(BinaryOperator<T> reducer) {
T tmpValue = this.list.get(0);
if(this.list.size() < 2) return tmpValue;
this.list = this.list.subList(1, list.size());
return reduce(reducer, tmpValue);
}
Optional<T> find(Predicate<T> predi) {
for(int i=0;i<this.list.size();i++) {
T value = this.list.get(i);
if(predi.test(value)) return Optional.of(value);
}
return Optional.empty();//편의상 null리턴
}
Integer findIndex(Predicate<T> predi) {
for(int i=0;i<this.list.size();i++) {
if(predi.test(this.list.get(i))) return i;
}
return -1;//편의상 null리턴
}
Boolean some(Predicate<T> predi) {
return findIndex(predi) != -1;
}
Boolean every(Predicate<T> predi) {
return findIndex(predi.negate()) == -1;
}
<R> List<R> pluck(String key,Class<R> typeToken){
Stream<R> result = map(val-> pluckHelper(val, key, typeToken));
return result.some(Objects::isNull).booleanValue() ? Collections.emptyList() :result.toList();
}
//런타임 시 타입이 확정되는 경우 어쩔 수 없이 Object를 리턴해야 한다.
<R> R pluckHelper(T val,String key,Class<R> typeToken) {
try {
Field field = val.getClass().getDeclaredField(key);
return typeToken.cast(field.get(val));
} catch (Exception e) {
return null;
}
}
List<T> toList(){
return this.list;
}
}
}
'개발 > 함수형 프로그래밍' 카테고리의 다른 글
자바스크립트 함수형 코드 자바화 해보기 - groupBy, countBy (0) | 2023.06.27 |
---|---|
자바스크립트 함수형 코드 자바화 해보기 - min, max, minBy, maxBy (0) | 2023.06.26 |
자바스크립트 함수형 코드 자바화 해보기 - find, findIndex, some, every (0) | 2023.06.24 |
자바스크립트 함수형 코드 자바화 해보기 - pipe, go 를 대체할 Stream 클래스 정의 (0) | 2023.06.23 |
자바스크립트 함수형 코드 자바화 해보기 - reduce (0) | 2023.06.22 |