이 프로젝트는 Robert C. Martin의 Clean Coders Screen Cast 중 Episode 10. OCP(Open but Closed Principle)에서 사용된 예제를 가지고 어떤 문제가 있는지, 어떻게 해결할 수 있는지를 짧은 변경 단위로 커밋하여 보여준다. 이 예제에서는
- Function Should Do One Thing
- SRP(Single Responsibility Principle)
- Feature Envy
- OCP(Open but Closed Principle)
등을 준수하는 방법을 아래와 같은 기법을 통해 보여준다.
- Extract Variable / Field / Method / Class / Interface
- Inline Variable
- Rename
- Change Signature
- Create a Field for Parameter
- Move Method
- Safely Delete
- Replace Type Code with Subclasses
- push members down
예제에서는 IDE로 intelliJ를 사용한다.
- 함수가 너무 크고,
- 중복도 존재(xx / 100)
- 작은 메소드들로 추출(extract)할 필요가 있다.
- Composed Method Pattern, Extract till drop 규칙에 따라 extract method를 수행
- 비즈니스 로직(비용 계산 로직)과 메시지/포맷팅 로직이 섞여 있다.
- 한가지가 아니라 여러가지 이유로 변경이 필요한 경우가 발생한다.
- 비지니스 규칙을 확장하고자 한다면 이 모듈을 수정해야 한다.
- 메시지와 포맷팅을 변경하고자 한다면 이 모듈을 수정해야한다(위 그림에서 붉은 박스처럼 타입에 의존한 코드로 인해)
만일 새로운 식사 타입으로 LUNCH를 추가하는 경우 타입에 의존하는 코드로 인해 어려움을 겪게된다. 시스템 내의 모든 소스에서 expense type에 의존하는 모든 switch-case나 if-else 문장을 수정해야 할 것이다.
- Expense의 type, amount 등을 이용하여 결정을 내리고 있다.
- 이런 로직은 Expense에 있어야 함.
다행히 테스트(명확한 문서로서 실행 가능한 요구사항 역할)는 존재하여 공격적으로 리팩토링을 할 수 있다.
Function Should Do One Thing을 준수하기 위해 extract method를 수행한다.
절차
- Extract Variable
- Extract Method
- 가독성을 높이고 Composed Method Pattern을 준수하기 위해 printHeader 메소드를 추출한다.
- 같은 이유로 printTotals 메소드를 추출한다.
- printExpenses 메소드를 추출하고 싶다. 하지만 이 로직은 비즈니스 로직과 섞여있다.
- 비즈니스 로직을 totalUpExpenses라는 메소드를 추출하고 싶으나 2개의 변수가 변경되어 extract할 수 없다.
*** 보통 코드의 여러곳에 사용되는 변수들은 리팩토링 전에 field로 추출하는 작업을 선행한다. 그럼 더 편하게 리팩토링이 가능하다.***
printReport에 전달된 printer가 여러 함수에 파라미터로 전달되고 있다. 이를 field로 생성하여 파라미터 개수를 줄인다.
- Create a Field for Parameter
- field로 변환한 printer를 파라미터로 갖는 메소드들에 대해 change signature를 수행하여 파라미터를 제거한다.
- {}를 없애기 위해 메소드 추출
- {}는 function should do one thing을 위반하고 있다는 증후이다.
- 불필요한 변수 선언을 제거하기 위해 name을 inline한다.
{}를 제거하기 위해 addTotals 메소드를 추출한다.
ExpenseReport에서 비즈니스 로직을 추출하여 이를 ExpeseReport라고 하기위해 지금의 ExpeseReport는 ExpeseReporter로 rename한다.
- extract class할 때는 관련된 메소드/필드를 함께 추출해야 한다.
- 사용하거나 호출하는 메소드/필드를 "Members to Extract"에서 잘 선택한다.
- extract class 후에 사용되지 않는 메소드들이 발생한다. 이를 제거한다.
- (intellJ에서는 커밋때 이런 warning이 있으면 알려주니… 커밋 전에 작업하는 것이 좋다.)
Feature Envy인 메소드 isOverage, isMeal를 Expense 클래스로 Move Method한다.
Type에 종속적이여서 OCP를 위반하고 있다. Replace Type Code with Subclasses를 적용하자.
- Test를 수정하여 Expense가 아니라 필요한 타입의 서브 클래스를 생성하도록 한다.
- IDE의 quick fix 기능을 이용하여 3개의 서브 클래스를 추가하고 테스트를 수행하여 성공을 확인
- 객체 생성시 파라미터로 전달하는 type(예. DINNER)과 클래스명(예. DinnerExpense)는 중복이다. 파라미터를 제거한다.
이제 새로운 타입이 추가되면 수정은 없다. 그저 Expense를 상속받는 새로운 클래스를 추가하면 된다.
하지만 ExpenseRepoter#getName는 새로운 타입이 추가될 때마다 수정되어야 한다. OCP 위반이다.
이를 새로운 클래스로 추출하고, 인터페이스를 추출하고, ExpenseRepoter가 인터페이스에 의존성을 갖게 하여 비즈니스 로직(app part)의 변경 없이 새로운 타입을 추가할 수 있도록 한다.
- Extract Class - ExpenseNamer
- 더 이상 사용되지 않는 ExpenseReport#getName 메소드를 safely delete한다.
- change scope of getName to public
- extract interface를 위해 변경한다.
- extract interface
이상의 과정에서 이와 같은 아키텍쳐를 얻었다.
- 붉은색으로 나눠진 윗 부분은 변경에 영향을 받지 않고 재사용될 수 있는 부분이고
- 아래 부분은 변경이 요구될 때 추가되어야하는 서브 클래스들과 변경되어야 할 구현체이다.
개인적으로는 getName을 Expese 클래스로 옮기면 더 좋을 것 같다는 생각이 든다.
- 클래스 다이어그램을 보면 복잡해 진 것 같으나, 기존 다이어그램에서 Expense가 data holder로서 객체의 역할을 수행하지 못하고 있는 것에 반해
- 리팩토링 후에는 책임(출력, 계산)이 분리되었고, Expense에 관련된 행위가 옮겨졌다. 또 다이어그램만으로 어떤 Expense들이 존재하는지 어떤 기능들이 있는지가 눈에 잘 들어온다.
- 또 데이터와 관련된 기능이 한 객체에 모이면서 내부의 변경이 외부에 영향을 주지 않게 되었다.
메일: [email protected] 이나 facebook design by object 그룹으로 의견이나 질문 받겠습니다.