Programing/Java

[자바8 람다의 힘] 4장 람다 표현식을 이용한 설계

Tomining 2016. 2. 10. 22:15
이 글은 "자바 8 람다의 힘" 책을 정리한 글입니다.

Chapter 4. 람다 표현식을 이용한 설계

앞에서 다른 내용들만 봐도 람다를 이용하여 코드를 좀 더 간결하고 읽기 쉽게 작성할 수 있다는 것을 알았다.
이 장에서는 람다 표현식을 사용하여 여러가지 패턴이나 설계들을 구현하는 방법에 대해서 알아보자.

1. 전략패턴(Strategy Pattern)

클래스를 생성하는 이유 중 하나는 코드를 재사용하기 위함이다. 좋은 의도이긴 하나 항상 그런 것은 아니다.

자산의 총합을 구하는 예제를 통해 클래스 내의 코드 재사용성을 향상시켜 보자.
먼저 Asset 클래스를 생성한다.

public class Asset {
public enum AssetType { BOND, STOCK };
private final AssetType type;
private final int value;

public Asset(final AssetType type, final int value) {
this.type = type;
this.value = value;
}

public AssetType getAssetType() {
return type;
}

public int getValue() {
return value;
}
}

간단하게 자산종류와 자산가치 라는 두 변수만 가지는 Asset 클래스를 생성했다.
이젠 자산의 총합을 개산해보자.

public class AssetUtil {
public static void main(String[] args) {
final List<Asset> assets = Arrays.asList(
new Asset(AssetType.BOND, 1000),
new Asset(AssetType.BOND, 2000),
new Asset(AssetType.STOCK, 3000),
new Asset(AssetType.STOCK, 4000)
);

System.
out.println("Total of all assets : " + totalAssetValues(assets));
}

public static int totalAssetValues(final List<Asset> assets) {
return assets.stream()
.mapToInt(Asset::getValue)
.sum();
}
}
Total of all assets : 10000

간단하게 Stream을 생성하여 합계를 계산하였다. 결과도 잘 나오는 것을 확인할 수 있다.
결과에서도 보듯이 동작에는 아무런 문제가 없다. 다만 여기에는 구조상 3가지 문제가 얽혀있다.

* iteration 을 어떻게하고
* 어떤 값들에 대한 합계를 계산하며
* 그 합계를 어떻게 구하는가

이렇게 뒤엉켜 있는 로직은 재사용성이 떨어진다.

만약 Bond(채권) 자산에 대해서만 합계를 구한다고 하면...

public class AssetUtil {
public static void main(String[] args) {
final List<Asset> assets = Arrays.asList(
new Asset(AssetType.BOND, 1000),
new Asset(AssetType.BOND, 2000),
new Asset(AssetType.STOCK, 3000),
new Asset(AssetType.STOCK, 4000)
);

System.
out.println("Total of bonds : " + totalBondsValues(assets));
}

public static int totalBondsValues(final List<Asset> assets) {
return assets.stream()
.filter(
asset -> asset.getAssetType() == AssetType.BOND)
.mapToInt(Asset::getValue)
.sum();
}
}
Total of bonds : 3000

Stock(주식) 자산의 총합을 구한다면 또 다시 totalBondsValues() 를 복사하여 filter에 있는 부분을 변경해줘야 한다.

public class AssetUtil {
public static void main(String[] args) {
final List<Asset> assets = Arrays.asList(
new Asset(AssetType.BOND, 1000),
new Asset(AssetType.BOND, 2000),
new Asset(AssetType.STOCK, 3000),
new Asset(AssetType.STOCK, 4000)
);

System.
out.println("Total of stocks : " + totalBondsValues(assets));
}

public static int totalBondsValues(final List<Asset> assets) {
return assets.stream()
.filter(
asset -> asset.getAssetType() == AssetType.STOCK)
.mapToInt(Asset::getValue)
.sum();
}
}
Total of stocks : 7000

총 자산, 채권, 주식 각각의 자산의 총합을 계산하는데, 유사한 코드가 너무 많이 생산되었다.

이를 해결하기 위해 전략 패턴이라는 것을 적용해보자.
(디자인 패턴의 종류로 자세한 내용은 디자인 패턴 책을 찾아보길 바란다.)

public class AssetUtil {
public static void main(String[] args) {
final List<Asset> assets = Arrays.asList(
new Asset(AssetType.BOND, 1000),
new Asset(AssetType.BOND, 2000),
new Asset(AssetType.STOCK, 3000),
new Asset(AssetType.STOCK, 4000)
);
System.
out.println("Total of bonds : " + totalAssetValues(assets, asset -> asset.getAssetType() == AssetType.BOND));
System.
out.println("Total of stocks : " + totalAssetValues(assets, asset -> asset.getAssetType() == AssetType.STOCK));
System.
out.println("Total of assets : " + totalAssetValues(assets, asset -> true));
}

public static int totalAssetValues(final List<Asset> assets, final Predicate<Asset> assetSelector) {
return assets.stream()
.filter(
assetSelector)
.mapToInt(Asset::getValue)
.sum();
}
}

totalAssetValues() 구조를 살펴보면,
filter() 를 통해서 합계 계산 대상을 정하고
mapToInt() 를 통해서 합계 대상 value를 추출하고
sum() 을 통해서 합산을 하게 됩니다.

즉, 합계 대상 객체 추출 -> 합계 value 추출 -> 합계 수행 순서로 진행됩니다.




2. Delegate

람다 표현식과 전략패턴을 활용하여 문제를 분리했다.
문제 분리를 클래스 레벨에서도 분리가 가능한데, 이 때 재사용 측면에서 보면 Delegation(위임)을 이용하면 더 좋다.

Delegation(위임) 을 람다표현식을 통해 구현해보자.

CalculateNAV 클래스를 사용하여 웹 서비스에서 받은 데이터를 사용하여 재무 연산을 해보자.

public class CalculateNAV {
private Function<String, BigDecimal> priceFinder;

public CalculateNAV(final Function<String, BigDecimal> priceFinder) {
this.priceFinder = priceFinder;
}

public BigDecimal computeStockWorth(final String ticker, final int shares) {
return priceFinder.apply(ticker).multiply(BigDecimal.valueOf(shares));
}
}

computeStockWorth() 메서드에서 priceFinder에  대한 주식시세표(ticker)를 요청하고 주식의 시세에 따라 기치를 결정한다.
이제 priceFinder가 필요하다.

예제에서는 Yahoo 주식시세표를 가져와 계산해 볼 예정이나, 먼저 Stub을 통해 TC를 만들어보자.

public class CalculateNAVTest {
@Test
public void computeStockWorth() {
final CalculateNAV calculateNAV = new CalculateNAV(ticker -> new BigDecimal("6.01"));
BigDecimal
expected = new BigDecimal("6010.00");

Assert.assertEquals(0,
calculateNAV.computeStockWorth("GOOG", 1000).compareTo(expected), 0.001
);
}
}

Stub 을 작성해서 주식시세표에 따른 가치를 계산해 보았다.
실제로 Yahoo주식시세표를 가져와서 계산해보자.

public class YahooFinance {
public static BigDecimal getPrice(final String ticker) {
try {
final URL url =
new URL("http://ichart.finance.yahoo.com/table.csv?s=" + ticker);

final BufferedReader reader = new BufferedReader(new InputStreamReader(url.openStream()));
final String data = reader.lines().skip(1).findFirst().get();
final String[] dataItems = data.split(",");
return new BigDecimal(dataItems[dataItems.length - 1]);
}
catch (Exception e) {
throw new RuntimeException(e);
}
}
}
@Test
public void computeYahooStockWorth() {
final CalculateNAV calculateNAV = new CalculateNAV(YahooFinance::getPrice);
System.
out.println(String.format("100 shares of Google worth : $%.2f",
calculateNAV.computeStockWorth("GOOG", 100)));
}
100 shares of Google worth : $54161.00

현재 구글의 한주 당 주식가치는 541.61$ 인 듯하다. 100주의 가치가 54161$로 계산되었다.



3. Decorate

앞에서 설명한 Delegation 은 뛰어난 기능이다. 여기에 Delegation 을 chain 으로 묶어서 Behavior 를 추가할 수 있다면 더 유용하게 사용할 수 있을 것이다.

다음 예제에서 람다표현식을 사용하여 Delegate 를 조합해보자.
(Decorator 패턴을 어떻게 사용하는지 알아보기 위한 예제이다.)

Camera 라는 클래스를 만들고 capture() 를 통해  Color 를 처리하는 필터를 만들어보자.

public class Camera {
private Function<Color, Color> filter;

public Color capture(final Color inputColor) {
final Color processedColor = filter.apply(inputColor);

return processedColor;
}
}

위와 같이 Camera 는 Function 타입의 filter 변수를 갖고 있고, 이는 Color 를 받아서 처리하고 결과로 Color 를 반환한다. 이는 하나의 필터만 사용하지만 다양한 필터를 적용할 수 있도록 수정해보자.
유연성을 얻기 위해 Java8에서 등장하는 Default Method 를 사용할 것이다.

쉽게 이야기하면 Interface 에 구현부가 있는 것인데, 추상 Method 에 덧붙여서 인터페이스는 구현 부분이 있는 Method 를 갖게되며, Default 로 마크된다. 이 Method 는 해당 인터페이스를 구현한 클래스에 자동으로 추가된다.

public class Camera {
private Function<Color, Color> filter;

public Color capture(final Color inputColor) {
final Color processedColor = filter.apply(inputColor);

return processedColor;
}

public void setFilters(final Function<Color, Color> ... filters) {
filter = Stream.of(filters)
.reduce((
filter, next) -> filter.compose(next))
.orElse(
color -> color);
}

public Camera() {
setFilters();
}
}

setFilters() 를 이용하여 각 필터를 iteration 하고 compose 를 통해서 chain 으로 연결한다.
만약 filter 가 없다면 Optional Empty 를 반환한다.

4. Default Method



5. 예외 처리