기본 콘텐츠로 건너뛰기

[자바 웹 프로그래밍]2장 문자열 계산기 구현을 통한 테스트와 리펙토링

이번엔 2장에 나와 있는 내용 정리와 느낀점을 정리 해 보겠다.

1. main() 메소드를 활용한 테스트의 문제점.
  - 소스코드 구현 후 정상적으로 동작하는지 확인 위해 일반적인 방법은 main()메소드를 활용하는 것이다.
  - 실제 서비스를 담당하는 프로덕션 코드와 이 프로덕션 코드가 정상 동작 하는지 확인을 위한 main() 으로 나뉜다.
  - 이 방법의 첫번째 문제점은 프로덕션코드와 main() 메서드가 함께 있다는 것이다.
  - 프로덕션 코드와 테스트코드(main)을 분리할 수 있다.
  - 두 번째 문제는 내가 구현하고 있는 메서드만 집중 할 수 없고, 클래스가 가지고 있는 모든 메서드를 테스트 할 수 밖에 없다.
  - 다른 문제는 항상 콘솔로 확인을 할 수 밖에 없다는 것이다.
  - 이를 위해 등장한 라이브러리가 JUnit 이다. 내 관심을 가지는 메서드에 대해 테스트 가능하다.

2. JUnit을 활용해 main() 메서드 문제 극복
2.1 한 번에 메서드 하나에만 집중.
  - JUnit관련 라이브러리 추가 후  테스트 메서드에 @Test를 붙이면 된다.
  - test 관련 코드 작성 후 Run > Run as> JunitTest를 실행해 보자.
  - 각각 테스트 메서드를 독립적으로 실행할 수 있기 때문에 현재 내가 구현하고 있는 프로덕션 코드의 메서드에 집중할 수 있다.
    
import org.junit.Test;

public class CalculatorTest {
    @Test
    public void add() {
        Calculator cal = new Calculator();
        System.out.println(cal.add(1,2));
    }

}

2.2 결과 값을 눈이 아닌 프로그램을 통해 자동화
    
import org.junit.Test;

import static org.junit.Assert.assertEquals;

public class CalculatorTest {
    @Test
    public void add() {
        Calculator cal = new Calculator();
        assertEquals(9,cal.add(3,6));
    }

}

  - JUnit은 assertEquals() 메서드를 제공. 첫번째 인자는 기댓값, 두번쨰 인자는 메소드 실행한 결과 값이다.
  - assertEqals()를 이용하면 실행 결과를 자동화 하는 것이 가능하다.
  - assertEquals(), assertTrue(), assetNull(), assertNotNull(), assertArrayEquals() 등이 있다.
  - @Befoe 애노테이션을 사용해 테스트 메서드 간 영향을 미치지 않으면서 독립적으로 테스트 메서드 실행이 가능하게 한다.
  - 메서드 실행마다 @Before가 실행한다.
  - @After도 마찬가지로 매서드가 실행될 때마다 메서드 후 @After가 호촐된다.

3 문자열 계산기 요구사항 및 실습
3.1 요구사항
  - 요구사항을 곧바로 구현을 시작하기 보다 구현 시작 전 작은 단위로 나누는 연습을 하는 것이 개발자의 역량을 키우는데 좋은 습관이다.
  - 프로덕션 클래스의 메서드를 여러개 생성하는 것이 아니라, 메서드 하나에 문자열 계산기의 모둔 요구사항을 구현해야 한다.
  - 여러 개의 메서드를 추가해야 하는 부분은 테스트 클래스에서 구현한다.

3.2 추가 요구사항   - 소스 코드를 완성했으면 반드시 뒤따라야 하는 과정이 중복제고, 읽기좋은 코드로 리펙토링이다.   - 리펙토링은 다음 요구사항에 맞춰 진행한다.     1) 메서드가 한 가지 책임만 가지도록 구현한다.     2) 인텐트(들여쓰기) 깊이를 1단계로 유지한다. (if, while문 등등)     3) else를 사용하지 마라.   - 세부 구현에 집중하도록 하지 않고 논리적인 로직을 쉽게 파악할 수 있도록 구현하는 것이 읽기 좋은 코드다.   - 요구샇항이 변경되면서 메서드 이름, 변수이름을 변경하는 것 또한 중요한 리펙토링이다.   - 가가 단계 개발 과정은 구현-->테스트-->리펙토링 순으로 진행한다.
 
import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class StringCalculator {
    int add(String text) {
        if(isNull(text)) {
            return 0;
        }

        int num = sumNumber(split(text));

        return num;
    }

    public String[] split(String text) {
        Matcher matcher = Pattern.compile("//(.)\n(.*)").matcher(text);
        if(matcher.find()) {
            String customMark = matcher.group(1);
            return matcher.group(2).split(customMark);
        }
        return text.split(",|:");
    }

    public int sumNumber(String[] values) {
        int sum = 0;
        for(String val : values) {
            int num = Integer.parseInt(val);

            isMinus(num);

            sum += num;
        }
        return sum;
    }

    public void isMinus(int num) {
        if(num<0) {
            throw new RuntimeException();
        }
    }

    public boolean isNull(String text) {
        return text==null || text.isEmpty();
    }
}
 
import org.junit.Before;
import org.junit.Test;

import static org.junit.Assert.assertEquals;

public class StringCalculatorTest {
    StringCalculator calculator;

    @Before
    public void init() {
        System.out.println("start init()...");
        calculator = new StringCalculator();
    }

    @Test
    public void addNull() {
        assertEquals(0,calculator.add(""));
        assertEquals(0,calculator.add(null));
    }

    @Test
    public void addOne(){
        assertEquals(3,calculator.add("3"));
    }

    @Test
    public void addShim(){
        assertEquals(6,calculator.add("1,2:3"));
    }

    @Test(expected = RuntimeException.class)
    public void addMinus() {
        calculator.add("1,-2");
    }

    @Test
    public void addMark() {
        assertEquals(5,calculator.add("//;\n2;3"));
    }
}


책 추천   - 테스트 주도 개발 : 고품질 쾌속 개발을 위한 TDD 실천법과 도구 1장 공개자료를 실습   - 테스트 주도 개발(켄트백 저/ 인싸이트)   - 리팩토링 : 코드 품질을 개선하는 객체지향 사고법(마틴 파울러)   - 4장까지 반드시 읽고, 나머지는 자신이 구현한 코드를 리펙토링하면서 참조하자. 2.5 정규 표현식   - 중요한 부분을 먼저 학습 후 문자열 조작이 많아지는 시점에 학습하도록 하자. 느낀점  이전까진 Test에 대해 귀차니즘과 빠른 개발을 위해 Test코드를 작성하지 않았었다. 당연히 리펙토링도 생각해 본적이 없는 것 같다. 리펙토링도 간단한 것 부터 차근차근 연습해 보면서 몸에 익히고 Test 코드도 습관화 되도록 해야 겠다. 그럼 자세한 소스는 https://github.com/jinioh88/JavaNextStep/tree/master/Test 여기를 참조. 그럼 끝~!!

댓글

이 블로그의 인기 게시물

[스프링부트2.0 낚시게시판] 01. 프로젝트 생성 및 환경을 세팅해 보자

첫번째. 이클립스에서 프로젝트를 생성해 보자. 빠밤! 1. 이클립스 실행하고 프로젝트 생성하기  - 이클립스 실행 후 File -> New -> Spring Starter Project클릭 ( 부트는 Spring Starter Project로!! )        해당 프로젝트 설정을 본인의 입맛에 맞게? 해주자. 처음엔 저와 똑같이 하는게 삽질(?)의 노고를 덜 수 있으니 저 같은 초보 개발자 분들이나 이제 막 공부를 시작 하셧다면 위와 가이 설정 하는걸 추천.  - New Spring Starter Project Dependencies    - Spring Boot Version : 2.0.2    - Core : DevTools, Security, Lombok 클릭    - Web : web  클릭    - SQL : JPA, H2 클릭    - template Engines : Mstache   --> 타임리프(요즘 회사에서 많이 쓴다고해서)   - 그다음 다음 -> Finish 클릭하게 되면 Maven에 관한 프로젝트가 생성된다.   처음엔 메이븐 디펜던시부분을 받느라 시간이 걸릴수 있다. 프로젝트 구조는 스프링과 별반 차이가 없어 보인다 프로젝트 구조 관련해선 조만간 포스팅 해봐야 겠다. (요즘 책 읽을 시간도 없어서...😂) 두번째. 실행을 해보자.  스프링 부트2.0의 특징은 자체적으로 톰켓이 내장 되어 있어 따로 톰켓을 설정하는 부분이 없어서 아주 매우 편안하게 되었다.    - 실행은 src/main/java 밑에 com.fishing.board 패키지 밑에 FishBoardApplication.java 오른쪽 클릭 후...

[Spring] 웹 개발에 필요한 기본 정보들

이번 시간에는 웹 어플리케이션 개발에 필요한 기본 정보를 알아보려한다. MVC 패턴과 이 내용들을 잘 다룰줄 안다면 기본적인 웹 동작은 구현할 수 있을 것이다. 웹 애플리케이션의 종류 화면으로 응답하는 웹 애플리케이션 웹 페이지 형태로 클라이언트에 응답한다. 데이터로 응답하는 웹 애플리케이션 사용자 인터페이스와 데이터를 분리해서 취급하는 애플리케이션은 JSON or Xml을 사용해 데이터 형태로 클라인언트에 응답한다. 애플리케이션 설정 서블릿 컨테이너 설정 스프링 MVC를 이용해 웹 애플리케이션을 할때 ContextLoaderListener, DispatcherServlet, CharacterEncodingFilter를 서블릿 컨테이너에 등록해야 한다. 애플리케이션 컨텍스트 설정 MessageSource와 PropertySourcePlaceholderConfigurer의 빈 정의도 필요하다. 프런트 컨트롤러 설정 자바 기반설정은 @EnableWebMvc를 추가하면 된다. XML에선 <mvc:annotation-driven>요소룰 추하가면 된다. @Controller 구현 컨트롤러에서 구현하는 처리 내용 선언형 처리 : 메서드 시그니처를 참조해 프런트 컨트롤러가 하는 일 프로그래밍형 처리 : 컨트롤러 클래스의 메서드 안에 하는 일 분류 처리내용 선언형 요청매핑 요청 데이터 취득 입력값 검사 수행 프로그래밍형 입력값 검사 결과 확인 비즈니스 로직 호출 이동 대상 확인 및 데이터 연계 이동 대상 지정 컨트롤러 클래스 작성 @Controller public class WelcomController {   } POJO 클래스에 @Controller를 지정하면 다음 효과를 얻을 수 있다. 컴포넌트 스캔 기능을 사용해 DI 컨테이너에 빈으로 등록할 수 있다. 요청을 처리하는 메...

[자바 웹 프로그래밍]1장 첫번째 양파껍질 벗기기

자바 웹 프로그래밍 Next Step 책을 가지고 스터디를 시작했다. 1장은 학습 방향과 학습법에 관한 얘기인데, 나에게 와 닿은 부분을 남겨본다. 웹 개발자 공부 순서  웹을 처음 접했을 때 어디서부터 어느것 부터 공부해야하고 어떤 책을 봐야할지 매우 막막했던 기억이 있다. 본 책에서는 HTML --> CSS --> 자바스크립트 --> 자바 --> 자바웹프로그래밍 --> DB  이 순서로 학습 방향을 제시하고 있다. 백앤드 개발자로 가고 싶어 Spring 프로젝트를 몇차례 진행한 결론은 자바스크립트의 흐름은 어느정도 알고 있는게 도움이 된다는 결론이다.  다음으로 소프트웨어 학습하는 좋은 방법 몇가지가 나오는데, 그 중 내가 최근에 똑같이 느꼈던 것이 있어 적어본다. '일단 무엇인가 만들어보는 경험을 한 후 이론적인 개념을 학습하고, 다시 다음 단계 경험을 하고 이론적인 개념을 학습하는 과정을 반복하는 것이라 생각한다' 라는 구절이 나오는데, 최근 몇차례 Spring 프로젝트를 하면서, 이론에 대한 대략적인것을 기반으로 수많은 삽질과 성취를 이루고 난 다음, 삽질했던 부분을 책이나 인터넷 자료를 봤을때 부족했던 부분이 '아 이건 이렇게 쓰는거구나' 라는 것이 확실히 느낄 수 있었다. 전에는 무조건 '이론먼저 완벽하게 하고 프로젝트 하자!' 식이었는데 이는 경험상 별로 효과를 보지 못한것 같다(이론 금방 까먹는다.. 왜 써야 되는지 못느껴서). 책을 통한 학습  다음 몇가지 책을 추천했는데, 내가 읽은것과 앞으로 읽어봐야 할것을 나열하겠다. 1. 열혈강의 자바 웹 개발 워크북 - 엄진영  이 책 나에게 웹의 세계로 인도를 해주셧던 엄진영 강사님 책이다. 웹이 어떻게 나왔는지 부터 스트링 사용까지 차례대로 차근차근 따라할 수 있게 아주 잘 설명되고 좋은 소스코드를 제공하고 있다. 웹 처음이라면 반드시 볼것을 추천한다. 2. SQL 첫걸음 3. 코딩을 지탱하는 기술...