Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[LBP] 정상화 사다리 1단계 제출합니다. #41

Open
wants to merge 4 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,9 @@ dependencies {
testImplementation platform('org.assertj:assertj-bom:3.25.1')
testImplementation('org.junit.jupiter:junit-jupiter')
testImplementation('org.assertj:assertj-core')
implementation 'ch.qos.logback:logback-classic:1.5.6'
implementation 'ch.qos.logback:logback-core:1.5.6'
implementation 'org.slf4j:slf4j-api:2.1.0-alpha1'
Comment on lines +17 to +19

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

로깅 라이브러리를 활용했네요!
로그가 필요했던 이유는 무엇인가요? 🤔

}

test {
Expand Down
14 changes: 14 additions & 0 deletions src/main/java/Application.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import model.Ladder;
import utils.ExceptionHandler;
import view.LadderView;

public class Application {
public static void main(String[] args) {
try {
Ladder ladder = Ladder.of(4, 4);
Comment on lines +7 to +8

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

4라는 원시값은 상수화하기에 충분한 근거가 있지 않았나요? 🤔
원시값 그대로 둔 이유가 궁금합니다!

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

4라는 원시값은 상수화하기에 충분한 근거가 있지 않았나요? 🤔 원시값 그대로 둔 이유가 궁금합니다!

원시값을 그대로 둔 이유는 없습니다. 말씀대로 상수화하기에 충분히 근거가 있었고, 단순 실수입니다. 마지막 미션임에도 불구하고 커밋 전에 다시 한번 살펴보는게 미흡했던 것 같습니다...ㅜ 신경쓰도록 하겠습니다 !!

LadderView.printLadder(ladder);
} catch (Exception e) {
ExceptionHandler.handleException(e);
}
}
}
17 changes: 17 additions & 0 deletions src/main/java/model/Ladder.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package model;

public class Ladder {
private final Lines lines;

public Ladder(Lines lines) {
this.lines = lines;
}

public static Ladder of(int width, int height) {
return new Ladder(LadderGenerator.generate(width, height));
}
Comment on lines +6 to +12

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

정적 팩토리 메소드 활용 좋습니다! 👍 다만, 정적 팩토리 메소드를 활용한 이상, 생성자는 private으로 닫아두어야 합니다.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

정적 팩토리 메소드 활용 좋습니다! 👍 다만, 정적 팩토리 메소드를 활용한 이상, 생성자는 private으로 닫아두어야 합니다.

정적 팩토리 메소드에 대해서 잘 몰랐었네요..
외부에서 생성자를 직접 호출 할 일 없이 정적 팩토리 메소드를 통해 간접적으로 객체를 생성하는데, 그럼 Ladder생성자가 외부에서 직접 생성될 일이 없을텐데요.. private이 당연하겠군요.. 알려주셔서 감사합니다 ^-^


public Lines getLines() {
return lines;
}
}
7 changes: 7 additions & 0 deletions src/main/java/model/LadderGenerator.java

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

다른 클래스의 메소드를 호출하고 그 반환값을 그대로 반환하는 메소드만 있는 클래스가 필요한가요?
이 클래스의 역할이 무엇인가요? 🤔

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

다른 클래스의 메소드를 호출하고 그 반환값을 그대로 반환하는 메소드만 있는 클래스가 필요한가요? 이 클래스의 역할이 무엇인가요? 🤔

Ladder 생성 -> LadderGenerator -> LineGenerator -> SingleLineGenerator 와 같은 단계로 Ladder가 구현됩니다.
중요한 것은 Lines와 Line입니다.

Ladder안에 Lines, Lines를 채우는 Line으로 그릴 수 있을 것 같은데요.
Lines를 생성하기 위해 Line을 생성하는 도메인 과정과, List을 일급컬렉션으로 관리하는 과정에서 LadderGenerator라는 필요 없는 클래스가 중간에 등장했습니다. Lines를 생성하는 것은 Ladder의 역할이라고 생각했거든요 !

이 답변을 작성하면서 코드를 다시 살펴보니, Ladder에서 Lines를 작성하는 것이 아니라, 이미 LineGenerator에서 Lines를 생성, 반환하고 있으므로 원래 계획인 도메인 절차가 이상했었습니다. LineGenerator를 LinesGenerator클래스명으로 변경하고, LadderGenerator를 삭제하여 필요없는 클래스를 정리하고 도메인 단계를 다시 설정 해주었습니다 !

Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
package model;

public class LadderGenerator {
public static Lines generate(int width, int height) {
return LineGenerator.generate(width, height);
}
}
32 changes: 32 additions & 0 deletions src/main/java/model/LadderValidator.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
package model;

import java.util.*;
import java.util.stream.*;

public class LadderValidator {
private static final Random RANDOM = new Random();

public static void validate(List<Line> lines, int width) {
IntStream.rangeClosed(0, width)
.forEach(i -> validateColumn(lines, i, width));
Comment on lines +10 to +11

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

forEach를 활용하려고 stream을 도입하는 것보다 for-loop가 훨씬 가독성이 높지 않을까요?
관련 아티클입니다! 😁

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

forEach를 활용하려고 stream을 도입하는 것보다 for-loop가 훨씬 가독성이 높지 않을까요?

말씀해주신 가독성은 for-loop가 훨씬 좋은 것 같습니다.

알려주신 아티클에서

pieces.keySet()
      .forEach(
           positionKey -> model.addAttribute(
               positionKey,pieces.get(positionKey)));

의 코드에서 혹시, forEach 내부에 로직이 하나라도 더 추가된다면 동시성 보장가독성 측면에서 문제가 생길 수 있다는 글을 발견했는데요.

for (String positionKey: pieces.keySet()) {
    model.addAttribute(positionKey, pieces.get(positionKey));
}

위의 코드 처럼, for-loop를 이용하여 수정한 후에, 로직이 추가된다면 forEach내부를 손봐야 하는 것이 아니라 filter나 map을 활용하거나 for-loop를 활용하는 것이 올바른 방향이라고 나와있기에 가독성 뿐만 아니라, 동시성과 성능면에서도 더 올바른 선택을 할 수 있을 것 같네요.

알려주셔서 감사합니다.

}

private static void validateColumn(List<Line> lines, int col, int width) {
boolean emptyColumn = lines.stream().noneMatch(line -> hasBridgeAt(line, col, width));

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
boolean emptyColumn = lines.stream().noneMatch(line -> hasBridgeAt(line, col, width));
boolean emptyColumn = lines.stream()
.noneMatch(line -> hasBridgeAt(line, col, width));

메소드 체이닝을 해야 할 때는 도트 연산자마다 개행을 해주는 편이 낫다고 생각합니다. 상화의 생각은 어떤가요? 🤔

if (emptyColumn) {
connect(lines.get(RANDOM.nextInt(lines.size())), col, width);
}
}

private static boolean hasBridgeAt(Line line, int col, int width) {
if (col == 0) return line.hasBridgeAt(col);
if (col == width) return line.hasBridgeAt(col - 1);
return line.hasBridgeAt(col - 1) || line.hasBridgeAt(col);
}


private static void connect(Line line, int col, int width) {
if (col == width) line.setBridgeAt(col - 1);
else line.setBridgeAt(col);

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

- else 예약어를 쓰지 않는다.

우리가 첫 미션부터 지켜야 했던 요구 사항을 위반했습니다.. 😢

}
}
23 changes: 23 additions & 0 deletions src/main/java/model/Line.java

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

왜 이 객체에서는 정적 팩토리 메소드를 활용하지 않았나요? 🤔

Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
package model;

import java.util.*;

public class Line {
private final List<Boolean> points;

public Line(List<Boolean> points) {
this.points = points;
}

public List<Boolean> getPoints() {
return points;
}

public boolean hasBridgeAt(int index) {
return points.get(index);
}

public void setBridgeAt(int index) {
points.set(index, true);
}
Comment on lines +20 to +22

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

외부에서 클래스 내부의 컬렉션을 변경하는 메소드가 있어도 괜찮은 걸까요? 🤔
이렇게 설계한 이유는 무엇인가요?

}
19 changes: 19 additions & 0 deletions src/main/java/model/LineGenerator.java

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Generator라는 클래스들이 등장하게 된 계기가 궁금합니다.
여기서 하고 있는 역할들은 일급 컬렉션들이 직접 해도 되지 않나요? 🤔

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

Generator라는 클래스들이 등장하게 된 계기가 궁금합니다. 여기서 하고 있는 역할들은 일급 컬렉션들이 직접 해도 되지 않나요? 🤔

말씀대로 일급 컬렉션들이 직접 해도 될 것 같아요. 원래, List<Line> lines = new ArrayList<>();

라는 코드를 Ladder클래스에 작성 했을 때에
일급컬렉션으로 따로 만들어야지
라고 생각해서...
Lines 클래스, Generator클래스 생성되고 정작 일급컬렉션 클래스에서는 get메서드만 있거나 도메인으로서 역할을 하지 못하고 있었네요..

멤버 변수 하나만 있으면 이미 일급 컬렉션이기에 도메인으로서 역할을 하지 못하는 클래스는 삭제하고 일급 컬렉션의 역할로 리팩토링 하였습니다.

감사합니다 !

Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
package model;

import java.util.*;

public class LineGenerator {

public static Lines generate(int width, int height) {
List<Set<Integer>> reserved = ReservedPositionGenerator.generate(width - 1, height);
List<Line> lines = new ArrayList<>();

for (int row = 0; row < height; row++) {
Line line = SingleLineGenerator.generate(width - 1, reserved.get(row));
lines.add(line);
}

LadderValidator.validate(lines, width - 1);
return new Lines(lines);
}
}
15 changes: 15 additions & 0 deletions src/main/java/model/Lines.java

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

getter만 존재하는 도메인(모델)이 도메인으로서 역할을 제대로 수행하고 있나요?
지금 이 클래스가 DTO와 무엇이 다른가요? 🤔

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

getter만 존재하는 도메인(모델)이 도메인으로서 역할을 제대로 수행하고 있나요? 지금 이 클래스가 DTO와 무엇이 다른가요? 🤔

맞습니다. 삭제하고 Lines도메인을 Ladder 멤버 변수로 옮겨줘서 쓸데 없는 도메인을 정리하였습니다 !

Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package model;

import java.util.List;

public class Lines {
private final List<Line> lines;

public Lines(List<Line> lines) {
this.lines = lines;
}

public List<Line> getLines() {
return lines;
}
}
50 changes: 50 additions & 0 deletions src/main/java/model/ReservedPositionGenerator.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
package model;

import java.util.*;
import java.util.stream.*;

public class ReservedPositionGenerator {
private static final Random random = new Random();

public static List<Set<Integer>> generate(int width, int height) {
List<Set<Integer>> reserved = initializeReservedList(height);
List<Integer> positions = createShuffledPositions(width);

assignPositions(reserved, positions);

return reserved;
}

private static List<Set<Integer>> initializeReservedList(int height) {
return IntStream.range(0, height)
.mapToObj(i -> new HashSet<Integer>())
.collect(Collectors.toList());
}

private static List<Integer> createShuffledPositions(int width) {
List<Integer> positions = IntStream.range(0, width)
.boxed()
.collect(Collectors.toList());
Comment on lines +25 to +27

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

IntStream이 필요한 로직이었나요? 🤔

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리뷰 감사합니다 !

IntStream이 필요한 로직이었나요? 🤔

필요 없습니다 ! depth를 신경쓰다보니 무조건 for, while 반복문 대신에 Stream을 사용하게 되었습니다.

forEach를 활용하려고 stream을 도입하는 것보다 for-loop가 훨씬 가독성이 높지 않을까요?
관련 아티클입니다! 😁

위 리뷰도 참고하여, 잘못된 습관을 가지고 있는 것을 알게되었습니다.
감사합니다 😆


positions.add(random.nextInt(width));

do {
Collections.shuffle(positions, random);
} while (isSequence(positions));

return positions;
}

private static boolean isSequence(List<Integer> numbers) {
return IntStream.range(0, numbers.size() - 1)
.anyMatch(i -> numbers.get(i).equals(numbers.get(i + 1)));
}



Comment on lines +42 to +44

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

무의미한 공백이 있네요.. 😢

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

무의미한 공백이 있네요.. 😢

앗.. 커밋 전 공백 및 라인 컨벤션에 더 신경쓰도록 하겠습니다 .... 😢

private static void assignPositions(List<Set<Integer>> reserved, List<Integer> positions) {
for (int i = 0; i < reserved.size(); i++) {
reserved.get(i).add(positions.get(i));
}
}
}
29 changes: 29 additions & 0 deletions src/main/java/model/SingleLineGenerator.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
package model;

import java.util.*;
import java.util.stream.*;

public class SingleLineGenerator {
private static final Random RANDOM = new Random();

public static Line generate(int width, Set<Integer> reserved) {
List<Boolean> points = new ArrayList<>(Collections.nCopies(width, false));

reserved.forEach(i -> points.set(i, true));

IntStream.range(0, width).forEach(i -> {
if (!points.get(i) && isNotOverlap(points, i)) {
points.set(i, RANDOM.nextBoolean());
}
});
Comment on lines +14 to +18

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Package java.util.stream Java 8 Stream 공식 문서에서는 다음과 같이 설명하고 있습니다!

Note also that attempting to access mutable state from behavioral parameters presents you with a bad choice with respect to safety and performance; if you do not synchronize access to that state, you have a data race and therefore your code is broken, but if you do synchronize access to that state, you risk having contention undermine the parallelism you are seeking to benefit from. The best approach is to avoid stateful behavioral parameters to stream operations entirely; there is usually a way to restructure the stream pipeline to avoid statefulness.

요약하자면, 병렬 Stream 사용 시 컬렉션의 상태를 동기화 하지 않으면 데이터 경합이 발생하고 그렇다고 상태를 동기화하면 병렬 처리의 이점을 모두 포기해야 합니다. 즉, Stream 연산 활용 시에는 상태를 가져선 안됩니다. 그런데, 현재 이 연산은 points의 상태를 변경시키고 있습니다. 병렬 Stream 연산에서는 문제가 생길 가능성이 높겠죠.. 😢


return new Line(points);
}

private static boolean isNotOverlap(List<Boolean> points, int i) {
if (i < points.size() - 1 && points.get(i + 1)) return false;
if (i > 0 && points.get(i - 1)) return false;

return true;
}
}
12 changes: 12 additions & 0 deletions src/main/java/utils/ExceptionHandler.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
package utils;

import org.slf4j.*;

public class ExceptionHandler {
private static final Logger logger = LoggerFactory.getLogger(ExceptionHandler.class);

public static void handleException(Exception e) {
System.err.println("시스템 오류 : " + e.getMessage());
logger.error("시스템 오류 : ", e);
}
}
31 changes: 31 additions & 0 deletions src/main/java/view/LadderView.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
package view;

import model.*;

public class LadderView {
private static final String BRIDGE = "-----|";
private static final String SPACE = " |";
private static final String BAR = "|";

public static void printLadder(Ladder ladder) {
printLines(ladder.getLines());
}

private static void printLines(Lines lines) {
lines.getLines().forEach(LadderView::printLine);
}

private static void printLine(Line line) {
System.out.print(BAR);
line.getPoints().forEach(LadderView::printPoint);
System.out.println();
}

private static void printPoint(Boolean point) {
if (point) {
System.out.print(BRIDGE);
return;
}
System.out.print(SPACE);
}
}