간단한 예제부터 시작해보자
TDD는 다른데서도 잘 설명되었지만 실패하고 통과하고 refactor하는 것이다.
이것의 반복형태다. 상당히 추상적이고 이게 뭐냐하것다. 나도 그랬으니까 ㅋㅋ
하나하나 코드로 만들면서 이해해보자~~ 유후~~
아래와 같이 win32 콘솔 응용 프로그램을 만든다. Project 명은 마음대로 만들어보자. ㅋㅋ
난 빈 프로젝트가 좋기 때문에 아래와 같이 빈 project를 만든다. ㅋㅋ
아래와 같이 소스파일에 cpp file 하나를 생성한다. 유후~~
아참 우리가 뭐할지 얘기를 안했구만. Code를 짠다는 것은 어떤 기능을 만드는 것이고 그 기능에 해당하는 method를 만드는 것이다.
내가 아는 대로는 그렇다. ㅋ
클래스를 만들고 그 클래스에 값들을 넣고 그 값들을 xml format 문자열 형태로 작성하는 것을 만들어보자. |
진짜 간단하다.
TDD 관련 Site를 뒤져보면 야구 게임/볼링 게임/은행 계좌 등등 샘플 소스들이 넘친다.
그런 분들 소스도 있으니 이런 소스도 만들어보면 어떨까? ㅋㅋ
그럼 우리가 만들 클래스는 학생 주소록 정도로 하자.
학생이라면 아래와 같은 정보들이 있것다.
이런것들이 있다.
이걸 간단한 xml format으로 만들어보자
<name>이원재</name>
<age>19</age>
<studentNumber>100979571</studentNumber>
<address>서울시 관악구 XXXX</address>
|
이정도면 우리가 TDD할 모든 준비가 끝났다. ㅋㅋ
이그림을 다시 보고 넘어가자.
- Write a test that fails
- Make the code work
- Eliminate redundancy
우리가 할 일은 실패할 Test Code를 먼저 짜야한다.
지금부터 나를 따르라. ㅋㅋㅋ
다음과 같이 코드를 짠다. 그런 다음 compile한다. 에러 안난다. ㅋㅋ 아무 것도 없으니 ㅋㅋ
다음과 같이 coding을 한다.
int main(void)
{
Student* s = new Student();
assert(NULL == s);
return 0;
}
|
그럼 아마 아래와 같이 붉은 줄들을 볼수 있다. 으미~~ ㅜㅜ
헉.. Student라는 Class를 만들지도 않았는데 객체를 생성한다.
그리고 assert라는 함수를 쓰는데 함수를 Header File을 include하지도 않았다.
이상태로 컴파일 해보자 ㅋㅋ
그럼 다음과 같이 에러를 뿜는다. ㅋㅋ
이게 바로 TDD의 첫번째 단계 Write a test that fails이다.
- Write a test that fails
- Make the code work
- Eliminate redundancy
그럼 이제 두번째 단계인 Make the code work 단계로 넘어가자
- Write a test that fails
- Make the code work
- Eliminate redundancy
우선 Class Student를 만들어보자. 다음과 같이 ㅋㅋ
class Student
{
};
int main(void)
{
Student* s = new Student();
assert(NULL == s);
return 0;
}
|
그런다음 컴파일을 한다.
그럼 다음과 같이 오류가 많이 준다. ㅋㅋ
이제 assert를 없에보자 . 다음과 같이 include를 해보자.
#include <assert.h>
class Student
{
};
int main(void)
{
Student* s = new Student();
assert(NULL == s);
return 0;
}
|
그래도 다음과 같이 나오네.. 뭘까.. ㅋㅋ
아 assert에는 NULL을 넣으면안되나 보다.. 헐 몰랐구만…
그냥 빼주자.
그럼 아래와 같이 해서 돌려보자. ㅋㅋ
#include <assert.h>
class Student
{
};
int main(void)
{
Student* s = new Student();
assert(s);
return 0;
}
|
그럼 아래와 같이 굿 오류가 없다. ㅋㅋ
그럼 두번 째 단계인 Make the code work는 성공적으로 pass했다.
ㅋㅋ
- Write a test that fails
- Make the code work
- Eliminate redundancy
이제 세번 째 단계인 Eliminate redundancy를 해야하는데.. 근데 refacoring할게 없다. 너무 simple해서
이건 이번단계에서 skip
그럼 다시 Write a test that fails를 만들어보자.
- Write a test that fails
- Make the code work
- Eliminate redundancy
학생들에게 필요한 요소인 아래 4개에 대한 Test Code를 만들어 보자.
각각의 학생들은 4가지가 필수잖아요? ㅋㅋ
그리고 입력할 값들의 예는 아래와 같아요.
<name>이원재</name>
<age>19</age>
<studentNumber>100979571</studentNumber>
<address>서울시 관악구 XXXX</address>
|
아래와 같이 코딩을 해봐요.
아까 우리가 짠 Test Code는 지우지 않기염 ㅋㅋ 계속 Append하면서 진행됩니다. ㅋㅋ
#include <assert.h>
class Student
{
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s->studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
다 작성했다면 Compile해요.
그럼 아래와 같이 또 오류가 발생합니다.
- Write a test that fails
- Make the code work
- Eliminate redundancy
이제 Pass하기 위해서 하나씩 오류를 제거해봅니다.
저 오류들만 제거하면 우리는 Coding완성이 되는거잖아요?!! ㅋㅋ
우선 생성자를 만들어서 Compile Error를 하나 잡아보아요.
ㅋㅋ
#include <assert.h>
class Student
{
public:
Student()
{
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s-> studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
그럼 아래와 같이 하나 줄였어요. ㅋㅋ
이제 또 하나씩 줄여보죠.
아래와 같이 Class에 변수들을 선언합니다. 이렇게 하면 개별 값들을 객체안에 저장할 수 있죠?!! ㅋㅋ
바로 컴파일 해보죠.
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
Student()
{
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s-> studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
보시면 또 오류가 줄어가고 있습니다.
ㅋㅋㅋ
이제 strcmp
함수에 대한 오류를 없에 보죠.
이건 include가 안되서 발생된걸꺼예요. ㅋㅋ
아래와 같이 코딩을 해보죠. 보이시죠.. #include<string.h>가 ㅋㅋ
#include <string.h>
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
Student()
{
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s-> studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
Compile 하면 아래와 같이 오류가 사라집니다. ㅎㅎ
그럼 이렇게 모든 오류를 잡았습니다. 그럼 debug mode로 돌려보죠. ㅋㅋ
F5를 누르자마자 요래 뜨네요. 허허
다시 시도를 눌러서 뭐가 문제인지 찾아보죠.
확인해보니 여기서 문제군요.
우리가 만든 객체에 우리가 넣어준 "이원재", 19, 100979571, "서울시 관악구 XXXX"가 안들어 있었어요.
그럼 들어가게 수정해보자구요.
오래 생성자에 함수를 추가합니다.
그리고 컴파일해봐요. ㅋㅋ
#include <string.h>
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
Student()
{
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
sprintf(Name,"%s",_Name);
age = _age;
studentID = _studentID;
sprintf(address,"%s",_address);
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s->studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
헉 sprintf
관련 오류가 발생하네요.
그럼 넣어줘야죠.
이렇게 stdio.h를 추가해봐요. 컴파일 오류도 안나고 assert 오류도 발생안되네요. ^^ 요후
#include <stdio.h>
#include <string.h>
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
Student()
{
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
sprintf(Name,"%s",_Name);
age = _age;
studentID = _studentID;
sprintf(address,"%s",_address);
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s->studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
이제 대망의 Refactoring 시간이 돌아왔습니다.
이제 Code를 이쁘게 해보죠. ㅋㅋ
- Write a test that fails
- Make the code work
- Eliminate redundancy
우선 Debug로 잡아보니 Class 객체의 Name[] 배열에 쓰레기 값들로 채워져 있네요.
깔끔하게 0로 초기화해보죠. ㅎㅎ
요래 coding해봤습니다.
memset으로 0초기화 ㅋㅋㅋ
그리고 student() 생성자도 동일하게 초기화 code를 넣었습니다.
이제좀 깔끔하네요.
#include <stdio.h>
#include <string.h>
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
Student()
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
age = 0;
studentID = 0;
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
sprintf(Name,"%s",_Name);
age = _age;
studentID = _studentID;
sprintf(address,"%s",_address);
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s->studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
return 0;
}
|
이 코드의 문제점은 class를 동적할당한 이후에 해제하는 code가 없었습니다.
이와 같이 delete code를 넣어줘서 메모리 누스를 또 막아보죠. ㅋㅋㅋ
#undef UNICODE
#include <stdio.h>
#include <string.h>
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
Student()
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
age = 0;
studentID = 0;
}
Student(char* _Name, int _age, int _studentID, char* _address)
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
sprintf(Name,"%s",_Name);
age = _age;
studentID = _studentID;
sprintf(address,"%s",_address);
}
};
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
delete s;
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s->studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
delete s;
return 0;
}
|
이제 파일을 좀 쪼개보자. 하나의 파일에 다 넣기에도 점점 코드량이 늘어나고 있다.
Class student는 cpp, header로 쪼개고
Test code는 main.cpp에 그냥 넣어놓자.
[student.h]
#undef UNICODE
#include <stdio.h>
#include <string.h>
#include <assert.h>
class Student
{
public:
char Name[1024];
int age;
int studentID;
char address[1024];
public:
Student();
Student(char* _Name, int _age, int _studentID, char* _address);
};
|
[student.cpp]
#include "student.h"
Student::Student()
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
age = 0;
studentID = 0;
}
Student::Student(char* _Name, int _age, int _studentID, char* _address)
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
sprintf(Name,"%s",_Name);
age = _age;
studentID = _studentID;
sprintf(address,"%s",_address);
}
|
[main.cpp]
#include "student.h"
int main(void)
{
// 1. Test 1번
Student* s = new Student();
assert(s);
delete s;
// 2. Test 2번
s = new Student("이원재",19,100979571,"서울시 관악구 XXXX");
assert(!strcmp(s->Name, "이원재"));
assert(s->age == 19);
assert(s->studentID == 100979571);
assert(!strcmp(s->address,"서울시 관악구 XXXX"));
delete s;
return 0;
}
|
요래 Test Code와 Class를 분리하니 태가 좀 난다. ㅋㅋ
난 sprintf를 좀 싫어해서 memcpy로 고쳐볼까 한다.
[student.cpp]
#include "student.h"
Student::Student()
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
age = 0;
studentID = 0;
}
Student::Student(char* _Name, int _age, int _studentID, char* _address)
{
::memset(Name, 0,sizeof(char)*1024);
::memset(address, 0,sizeof(char)*1024);
memcpy(Name,_Name,strlen(_Name)*sizeof(char));
age = _age;
studentID = _studentID;
memcpy(address,_address,strlen(_address)*sizeof(char));
}
|
변경점이 있을때마다 F5를 눌러주라. 그럼 알아서 Test를 해준다. ㅋㅋ
바로 이게 좋은 것중 하나다. 내가 변경한것들을 바로바로 Test할 수 있다는 것 TDD의 최대 장점이다.