-
Notifications
You must be signed in to change notification settings - Fork 33
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
base: main
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
원시값을 그대로 둔 이유는 없습니다. 말씀대로 상수화하기에 충분히 근거가 있었고, 단순 실수입니다. 마지막 미션임에도 불구하고 커밋 전에 다시 한번 살펴보는게 미흡했던 것 같습니다...ㅜ 신경쓰도록 하겠습니다 !! |
||
LadderView.printLadder(ladder); | ||
} catch (Exception e) { | ||
ExceptionHandler.handleException(e); | ||
} | ||
} | ||
} |
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 정적 팩토리 메소드 활용 좋습니다! 👍 다만, 정적 팩토리 메소드를 활용한 이상, 생성자는 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
정적 팩토리 메소드에 대해서 잘 몰랐었네요.. |
||
|
||
public Lines getLines() { | ||
return lines; | ||
} | ||
} |
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 다른 클래스의 메소드를 호출하고 그 반환값을 그대로 반환하는 메소드만 있는 클래스가 필요한가요? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
Ladder 생성 -> LadderGenerator -> LineGenerator -> SingleLineGenerator 와 같은 단계로 Ladder가 구현됩니다. Ladder안에 Lines, Lines를 채우는 Line으로 그릴 수 있을 것 같은데요. 이 답변을 작성하면서 코드를 다시 살펴보니, 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); | ||
} | ||
} |
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
말씀해주신 가독성은 알려주신 아티클에서
의 코드에서 혹시, forEach 내부에 로직이 하나라도 더 추가된다면
위의 코드 처럼, 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)); | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
메소드 체이닝을 해야 할 때는 도트 연산자마다 개행을 해주는 편이 낫다고 생각합니다. 상화의 생각은 어떤가요? 🤔 |
||||||||
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); | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. - else 예약어를 쓰지 않는다. 우리가 첫 미션부터 지켜야 했던 요구 사항을 위반했습니다.. 😢 |
||||||||
} | ||||||||
} |
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 외부에서 클래스 내부의 컬렉션을 변경하는 메소드가 있어도 괜찮은 걸까요? 🤔 |
||
} |
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
말씀대로 일급 컬렉션들이 직접 해도 될 것 같아요. 원래, 라는 코드를 Ladder클래스에 작성 했을 때에 멤버 변수 하나만 있으면 이미 일급 컬렉션이기에 도메인으로서 역할을 하지 못하는 클래스는 삭제하고 일급 컬렉션의 역할로 리팩토링 하였습니다. 감사합니다 ! |
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); | ||
} | ||
} |
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
맞습니다. 삭제하고 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; | ||
} | ||
} |
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 리뷰 감사합니다 !
필요 없습니다 ! depth를 신경쓰다보니 무조건
위 리뷰도 참고하여, 잘못된 습관을 가지고 있는 것을 알게되었습니다. |
||
|
||
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 무의미한 공백이 있네요.. 😢 There was a problem hiding this comment. Choose a reason for hiding this commentThe 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)); | ||
} | ||
} | ||
} |
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Package java.util.stream Java 8 Stream 공식 문서에서는 다음과 같이 설명하고 있습니다!
요약하자면, 병렬 Stream 사용 시 컬렉션의 상태를 동기화 하지 않으면 데이터 경합이 발생하고 그렇다고 상태를 동기화하면 병렬 처리의 이점을 모두 포기해야 합니다. 즉, 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; | ||
} | ||
} |
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); | ||
} | ||
} |
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); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
로깅 라이브러리를 활용했네요!
로그가 필요했던 이유는 무엇인가요? 🤔