블로깅은 힘들어! ㅠㅠ

지금 블로깅을 좀 대충 하고 있는데, 그 이유는 뭐랄까 블로그 포스팅은 뭔가 전문인이 쓴 것처럼 잘 써야겠다는 강박 관념 같은 것이 있어서 글의 완성도를 높이려다보니 오히려 아무 글도 쓸 수 없는 지경이 되었기 때문이다. 이것 저것 조금씩 써놓은 것은 있는데 공개를 하지 못하고 있다.

게다가 방문자 수도 눈에 들어오는지라 현재 일주일에 들어오는 사람이 한자릿수이니 의욕이 떨어진다. 눈물 좀 닦고...ㅠㅠ


위키와 블로그?

그래서 생각해본게 위키이다. 난 회사에서 적극적으로 위키를 사용하고 있다. 처음엔 잘 몰랐는데, 지식을 점점 누적, 발전시켜서 글을 성장시킬 수 있는 특징이 있다.

생각해보면 위키에 글을 쓰는 것과 내가 블로그에 공개를 하지 않고 글을 쓰는 것과는 공통점이 있다. 둘 다 글을 발전시켜 나갈 수 있다는 것이다. 그런데 이상하게도 위키는 계속해서 글이 발전해 나가고, 블로그에서는 글을 잘 쓰지 않게 된다. 왜 그럴까? 아마도 블로그는 글을 쓸 때 주제를 먼저 정해서 쓰는 면이 있지만, 위키는 모든 지식을 덤프하듯 쏟아낸 후 잘 분류하고 가꾸기 때문일 것이다.

블로그에서도 이런 것이 아주 불가능한 것은 아니지만, 도구에서 느껴지는 '아주 약간의 차이'가 이런 결과를 만들었다. 위키는 위키 문법이란게 있어서 글을 꾸미는 것 보다는 빠르게 구조화하고 확장하고 리팩토링하는데 유용하도록 만들어졌기 때문이다.


그 밖에도 어떤 기사 중에 지식 관리는 위키로 하고, 위키 문서가 잘 정돈되면 그것을 블로그로 발행하는 것이 대세라는 기사를 보고 마음을 굳혔다. 나도 위키랑 블로그 둘다 해보는거야![각주:1]

그래서 개인적으로 로컬 머신에 위키를 만들어 두려고 했었다. 외부에 공개하지 않고 개인적으로만 메모장처럼 쓸 수 있는 그런 위키 말이다. 


스프링노트와의 만남

그 과정에서 당면한 첫 번째 문제는 '어떤 위키를 쓸까' 였다. 위키도 종류가 많아서, DBMS를 쓰는 위키도 있고 파일 시스템을 사용하는 위키도 있고 각 위키별로 문법 또한 다양하다.

조사를 하다 보니 스프링노트라는 것이 눈에 띄었다. 웹에서 몇 번 보기는 했지만 그저 그런 서비스라고만 생각하고 있었다. 그런데 스프링노트가 위키랑 비슷하다고 한다! 글을 '싸고' '링크를 따고' '리팩토링'할 수 있는 서비스인 것이다. 이걸 왜 이제서야 알았을까.

특히 kkamagui님께서 본인은 스프링노트와 티스토리를 사용한다고 하여 나도 한번 해보기로 했다.


지금 스프링노트를 이틀째 사용하고 있는데 상당히 만족하고 있다. 위키에 좀 익숙한 편이라 서비스를 이해하고 적응하는데는 별로 어려움이 없었다. 

웹 환경임에도 불구하고 워드프로세서와 비슷한 GUI를 제공한다는 점이 마음에 들었다. 단축키 지원도 괜찮은 편이다. 그리고 모든 글을 비공개로 해둘 수 있어서 개인적으로 사용하는데도 문제가 없다. 게다가 인터넷이 가능한 곳이면 어디서든 접근할 수 있는 것도 상당히 뛰어난 장점이다.[각주:2]

안타깝게도, 스프링노트는 2012년 9월 서비스 종료가 되고 말았다.


  1. 사실 여기서 고민을 좀 했었다. 위키와 블로그 모두 동일한 내용이 담기면, '중복'이 발생하는 것이다. 알다시피 중복은 아주 나쁜 것이기 때문에 둘 다 사용할지 말지에 대해 많이 고민을 했었다. [본문으로]
  2. 다만 접근성이 뛰어난 만큼 보안에도 유의를 해야겠다 [본문으로]
반응형
회사에서 테스트 주도 개발(원제: Test Driven Development: By Example) 2판을 빌려와서 설 연휴에 보고 있다. TDD에 대해 권위있는 책이다. 처음 보는 것은 아니지만 요즘 TDD가 잘 안되서 기초를 다지기 위해 다시 한번 보고 있다.

그런데 문제 - 사실 나쁜 의미에서의 문제는 아니다 - 는 모든 코드가 Java로 짜여져 있다는 것이다. 그리고 나는 C++에서 적용하고 싶었다. 내가 이 책을 처음 보았을 때는 TDD에 대해 아무 것도 모르던 시절이라 그냥 이런 것이 있나 보다 하고 눈으로 따라가면서 읽은 것이 다였다. 원래는 키보드로 쳐가면서 했어야 했지만 그러지 못했다. 그 이유 중에 하나가 바로 C++에서 사용할 수 있는 테스팅 프레임워크를 잘 몰랐던 것이다.

지금은 GoogleTest를 몇 달 동안 사용해왔기 때문에 잘 따라갈 수 있을 것 같다.


Money 객체를 만들면서 예제를 따라가다보니 언어의 차이 때문에 몇 가지 문제가 발생했다.

첫번째는 비교 구문이다. 
Java에서는 equals()를 사용하기 때문에 operator==()로 대신 구현했다. 그런데 중간에 보면 Dollar와 Franc을 상위 클래스에서 비교하는 구문에서 RTTI를 이용하는 부분이 나온다. Java에서는 getClass()로 런타임에 객체의 타입을 알아낼 수 있다. 그래서 책에서는 통화가 다른 경우 false를 리턴하게 했다. C++에서도 typeid()를 사용할 수 있기는 한데 이상하게 제대로 동작하지 않았다. 그래서 어쩔 수 없이 해당 부분 테스트는 주석 처리해버렸다. 이것 때문에 뒤의 몇 가지 테스트는 제대로 돌려보지 못했다.

두번째는 팩토리 패턴을 구현하는 부분에서부터 발생했다.
Money 클래스에서 자식 객체인 Dollar를 리턴하는데 코드가 아래와 같았다.
Money five = Money::Dollar(5);
그런데 Money 클래스는 abstract class라서 인스턴스화가 되지 않는다는 컴파일 에러가 떠버렸다. 아악... 이 문제 때문에 잠시동안 C++ 언어를 욕하다가 그냥 shared_ptr를 사용하는 것으로 문제를 해결했다.
typedef ::std::tr1::shared_ptr<Money> MoneyPtr;
MoneyPtr five = Money::Dollar(5);
물론 거의 절반에 가까운 테스트 코드와 클래스 코드를 바꾸어야 했다. 가끔 코드를 짜다 보면 이렇게 막다른 골목에 부딪히는 경우가 있다. 나 같은 경우 C++에서는 포인터 사용을 최대한 하지 않으려고 하는 경향이 있는데 그러다 보니 이런 문제가 가끔씩 생긴다 - 즉 포인터를 써야만 하는 상황이 오고야 말았을 때 시그니처가 바뀌면서 코드가 뒤집힌다.

문제는 한 번 스마트 포인터를 사용하기 시작하면 코드 전체가 스마트 포인터를 써야 한다는 것이다. 메소드의 파라미터, 리턴값, 생성자에서 받을 포인터와 객체 내에 보관하는 포인터까지... 결국 모든 클래스의 인스턴스를 스마트 포인터로 처리했다. 

개인적으로 스마트 포인터를 쓰는 것은 나쁘지 않다고 생각하지만, 문제는 코드 전반에 스마트 포인터에 종속성이 걸리는 경우다. 후반부로 갈 수록 거의 돌이키기 힘든 디자인 문제가 되기 때문에 아주 골치가 아프다.

반응형
알고리즘 문제 사이트들이다.

UVa Online Judge

내가 요즘 가는 곳. 그런데 엄청 페이지 로딩 속도가 느리다. 심각할 정도.

가입하고 Online Judge의 Browse Problems에서 문제를 골라 소스를 작성한 뒤 제출하면 정답 여부를 확인할 수 있다. 

서버의 컴파일러는 gcc이므로 Visual C++ 컴파일러로 잘 돌아가는 것도 업로드하면 컴파일 에러 나는 경우가 있으니 주의한다.
예를 들면 템플릿 사용할 때 꺾쇠를 연이어서 사용하면 - << 나 >> - VC++에서는 잘 인식되지만 gcc에서는 스트림 연산자로 인식되므로 한 칸 띄어써야 한다. 


USACO

대학교 다닐 때 ACM 모임 들어가서 처음으로 알게 된 알고리즘 문제 사이트. UVa에 비해 페이지가 좀 촌스럽지만 속도가 잘 나오는 편이다.

2010-11-04 
한번 방문해봤는데, 2007년 3월에 다섯 문제 풀고 한 문제 본 상태이다. 좀 부끄럽다. 당시 몇 번 나가다가 학업이 바쁘다는 핑계로 나가지 않았는데 지금 생각하니 좀 후회가 된다. 계속 했으면 지금의 내가 어떻게 변해 있었을까?



알고리즘 문제는 공학이라기 보다는 수학의 영역에 가까워서, 코드를 보기 좋게 한다거나 구조나 설계 같은걸 잘 만드는 것 보다는 문제를 빨리, 그리고 정확하게 푸는 게 더 중요하다.

풀다 보면 두 가지 상황이 발생하는데, 하나는 공학 - 그러니까 프로그래밍 - 스킬의 부족으로, 문제를 푸는 방법은 남에게 말할 수 있는데 코드로 작성하는 못하는 상황이다.

다른 하나는 흔한 상황으로, 그냥 푸는 방법을 찾지 못하는 경우이다.


문제를 풀다 보면 지루하고 귀찮고 짜증나서 집중하지 못하고 웹서핑을 한다거나 게임을 켜버린다거나 하는데, 어떻게 해야 극복할 수 있을 지 생각해보고 있는 중이다. 마치 숙제를 하는 기분이다.

사실 산수 문제 푸는걸 어렸을 때부터 별로 좋아하지 않았다. 과학은 아주 좋아했지만, 수학은 곱셈 배울 때 부터 나를 힘들게 했다. 어쩌면 구구단을 맞으면서 배워서 그럴지도 모르겠다.



반응형
요즘 키보드 앞에서 멍하니 있는 경우가 많아졌다.

실제로 멍하니 있는 것은 아니다. 머릿속에서는 무엇인가를 끊임없이 생각하고 있다.

간단한 모듈을 작성할 때도 종속성과 인터페이스, 호출 관계 등 수많은 조건과 변수들이 머릿속에서 구성이 된다. 그러면 좋은거 아니냐? 라고 생각할 수도 있겠지만, 그게 그렇지가 않다.

머릿속의 용량이 작은 관계로 아주 빠르게 그 내용을 잃어버린다. 말하자면 설계도를 그리고 있는데, 좀 그리도 보면 앞에 그렸던게 사라져있는 것이다. 그러면 앞부분을 다시 그리다 보면 뒤 부분이 사라져있고...

그래서 계속 뺑뺑이를 돌면서 생각을 하게 된다. 했던 생각 또하고 또하고. 그러다 보면 금방 지쳐버린다.

일단 코드에 싸지르고나서 리팩토링을 하는게 더 좋을 것 같다. TDD가 그런 점에서 조금 도움을 줄 수 있을 것이라고 기대를 했지만, 실천법이 미묘하게 잘 지켜지고 있지 않은 것인지 일단 테스트를 쓰고 통과하는데 집중하기 보다는 테스트 만들때부터 너무 생각이 많이 들어가는 것 같다.


내가 잠이 많은 이유도 평소에 생각이 너무 많아서인 것 같다. 눈 뜨고부터 잠들 때까지 항상 머릿속에서는 무슨 생각인가를 하고 있다. 이런 저런 생각이 계속 나서 잠도 쉽게 들지 않는 편이다. 심지어는 잘 때조차 항상 꿈을 꾼다. 꿈이 기억나지 않아야 깊이 잠든 것이라고 하는데 나는 깊이 잠이 들지 않는 모양이다.

아는 사람 중에는 눕자마자 코를 골면서 자는 사람도 있는데 정말 부럽다 ㅠㅠ

반응형
회사에서 메신저로 친구와 이야기 하다가 친구가 메신저로 웹페이지 링크 하나를 보내줬다.

클릭해보니까, 내용은 어떤 온라인 게임 상에서 남자가 여자인 척 하고 - 일명 넷카마 - 다른 여자와 아주 친해져서 결국 만남을 가지게 되었는데 알고 보니 그 상대도 남자더라는 이야기다. 여기까진 좀 웃기면서도 슬픈 이야기였다.

그런데 다른 링크를 보내줬는데, 아까 글을 썼던 사람이 다시 쓴 글이다. 상대방이 자기보고 사기죄로 고소한다는데 어떻게 해야 하냐고 한다. 참고로 글쓴이는 자신이 이십대 초반이라고 했다.


몇 가지 생각이 떠올랐다. 일단 사기죄는 성립될 수 없다는 생각이었다. 상대방은 '네가 여자인 척 해서 온라인 게임상의 아이템이나 사이버 머니를 받아냈으니 사기다', 또는 '남자인데 여자 행세를 했으니 사기다'라는 주장을 한 것으로 보인다. 

물론 게시물에는 이런 내용이 적혀있지는 않았지만 여케에게 남자가 경제적인 지원을 많이 하는 게 꽤 흔한 것으로 알고 있다. 하지만 이 사람은 악의적으로 속이려고 한 것도 아니고, 여케라는 이유로 남자가 금품을 제공하는 건 남자의 자의에 의한 것으로 볼 수 있으므로 성립이 안 될 것이다.

둘째로 남자가 여자 행세를 했다는 거 자체가 잘못되었다는건... 뭐 그냥 그럴수도 있지. 롤플레잉 게임인데 여케면 당연히 여자 노릇을 해야 맞지 않겠습니까(?!).


위는 아주 잠시의 순간에 생각한 것이고, 그 다음으로 생각한 것은 그렇게 벌벌 떨지 말고 법전을 보면 되지 않을까 하는 생각이었다. 그런데 이십대 초반의 남자가 이 정도 협박에 떨어서야... 아무튼 대법원 홈페이지에 들어가서 이곳 저것 뒤지기 시작했는데... 아무리 봐도 법전은 어디서 보는지 모르겠다. 뒤지다 보니 판례랑 법령이라는 것을 조회할 수 있는 곳은 찾았는데 법전이랑은 다른지 사기죄같은건 검색이 안된다.

에라이~

그런데 네이버에서 법전을 검색하다보니...

법전 [法典, code]


법전을 편찬하다

compile law books


법전을 제정하다

establish a code

출처: 네이버 사전


오오 이럴수가! 법전이 영어로 '코드'였다. 게다가 컴파일이라는 단어까지 보인다. 그렇다면 우리는 말하자면 국회의원인가? 아니면 법관 정도 되려나.


사실 법전이 실제 소스 코드와 비슷한 점이 많다고 평소에 생각하긴 했다. 그래서 애매한 법 때문에 말도 안되는 판결이 나오는 것을 보면서 버그와 비슷하다고 생각하기도 했고 - 정확히 말하면 요구사항이 애매한 것 때문에 코드가 의도와 다르게 나오는 것과 같다 -, 숙련된 프로그래머가 법을 작성하면 훨씬 더 잘 할수 있을 것 같다는 생각도 해봤었다.

한편으로는 국회의원이라고 해도 4천만 국민의 모든 상황에 맞게 법을 만드는 것도 참 힘들 것 같다는 생각도 들고, 코드가 늘어나니 버그도 생기고 복잡도도 늘어나는 것은 당연하다는 생각도 든다.


이것도 전공 강박증인가?

반응형

'일기' 카테고리의 다른 글

위키 찾다가 스프링노트를 찾았다  (0) 2011.02.15
코딩 슬럼프  (0) 2010.11.02
Test Driven Development, 첫 걸음부터 세미나를 하기까지  (0) 2010.10.10
레거시 코드와 씨름 중  (0) 2010.10.10
프로그래밍 심리학  (0) 2010.09.15
아래 코드를 보자.

typedef unsigned int UINT32;

std::string text = GetText();
for(UINT32 i = text.size() - 1; i >= 0; --i)
{
    // 텍스트를 뒤에서부터 읽어서 처리
    DoSomething(text[i]);
}
주석처럼 스트링을 뒤에서부터 읽어가면서 뭔가 처리를 하려고 한다. 그런데 프로그램이 뻗어버린다.

이유는 unsigned 때문이다. i가 0까지는 잘 내려오지만, 그 직후에 언더플로우가 발생해버린다. 그러면 약 42억의 수를 가지게 될 테고... 결과는? 쾅~


int를 사용하면 되겠지만, 저기 std::string::size() 라는 녀석은 리턴값이 size_t이다. 즉 unsigned int와 같다. unsigned와 int를 섞게 되면 컴파일러가 경고를 하게 된다. 경고는 당연히 피해야 하는 것이고 따라서 위와 같이 unsigned를 사용했던 것이다.



위 코드에서는 STL 자료구조가 사용된 관계로, 그냥 reverse_iterator를 사용하면 해결된다. 하지만 index를 알아야 하는 경우가 있다면... 어떻게 해야 할까?

내 생각엔 int i = static_cast<int>(text.size() - 1) 처럼 사용하는게 낫지 않나 싶다. 가독성이 좀 떨어지는 단점이 있지만 딱히 방법이 없다.

이럴 땐 Java처럼 unsigned를 안쓰는 정책을 사용하거나, C#처럼 웬만하면 기본 라이브러리는 int형을 최대한 사용하고 signed와 unsigned가 같이 쓰이는 경우 컴파일 에러(심지어 케스팅을 해도)를 발생시켜주는 언어가 약간 부럽다.


반응형
좀 전에 아래와 같은 코드를 작성하고 있었다.[각주:1]

using ::std::string;
using ::std::deque;

bool Slump::Parse(const string& string, deque< string >& stringsOut)
{
    return true;
}


그런데 컴파일 에러가 발생했다. deque의 _Ty 템플릿 인수가 잘못되었다고 한다. 대체 뭐가 문제인지 1분동안 고민했다. 그리고 알아냈다.

0번 파라미터 이름이 string이라서, 1번 파라미터를 만들 때 deque의 템플릿 파라미터로 넣은 string이 타입이 아니라 변수로 인식된 것으로 보인다. 0번 파라미터 이름을 str 등으로 바꾸거나, 1번 파라미터를 deque<std::string> 으로 변경하자 문제가 사라졌다.

그러니까 0번 파라미터의 변수가 1번 파라미터에서 사용되어버렸다는 소린데... 컴파일러가 무슨 생각으로 이렇게 동작하는지 모르겠다. 파라미터 간 데이터 전달이라도 가능한 문법이라도 있는걸까?

한가지 더 테스트해보았다.

using ::std::string;

void Foo()
{
    string string;
    string stringAnother;
}
여기서도 첫번째 string string은 잘 컴파일되지만, 두번째 string stringAnother는 컴파일 에러가 발생한다.



보통 아직 의미가 확정되지 않은 - 예를 들면 파싱할 때 입력되는 스트링 - 스트링 변수는 그냥 이름을 string으로 짓는데, 저런 경우에 혼란을 발생시킬 문제가 있으니 다른 것으로 바꿔야겠다.

그런데 뭘로 하지... text? str? s?

코드는 최대한 '소리내어 읽을 수 있는' 형태로 작성하고 싶기 때문에 살짝 고민이 된다.

그냥 타입은 std::string으로 써버릴까?
그러자니 가독성이 저해된다. 그냥 text로 쓰는게 그나마 제일 나을 것 같다.

  1. SyntaxHighlighter에서 꺾쇠 기호를 사용하면 태그로 인식되기 때문에 string 좌우로 꺾쇠에 공백을 넣었다. 원래 코드를 작성할 때는 꺾쇠에 공백을 넣지 않는다. [본문으로]
반응형
사실 이미 인지하고 있었던 문제였긴 한데, 최근에 몇 번 또 당해버렸다.

GoogleTest를 사용하면서 라이브러리를 미리 빌드해놓고 쓰는데 디버그 모드로 빌드한 것과 릴리즈 모드로 빌드한 것 두 가지로 만들고, 디버그 모드로 빌드한 것은 lib 파일 이름 뒤에 'd'를 붙여서 사용하고 있다.

현재 작성중인 코드를 디버그 모드로 빌드할 때는 GoogleTest 라이브러리도 디버그 모드로 빌드한 것을 써야 하고, 릴리즈 모드로 빌드할 때는 라이브러리 역시 릴리즈 모드로 빌드한 것을 써야 한다.

만약 이렇게 하지 않으면 컴파일은 잘 되는데 런타임에 프로그램이 에러를 뱉고 죽어버린다.
그 외에도 std::map을 넘겨받았는데 내용물이 완전 엉망이 되어서 받은 적도 있다.




그런데 CRT를 맞춰봐도 역시 동일하게 프로그램이 죽어버린다. 그러니까 GoogleTest의 라이브러리와 내 작업중인 프로젝트의 프로젝트 속성-C/C++-코드 생성-런타임 라이브러리를 "다중 스레드 DLL(/MD)"로 통일했는데도 계속 발생한다는 말이다.

그래서 이번엔 CRT를 맞춰준 상태에서 Preprocessor의 _DEBUG를 NDEBUG로 바꿔봤는데, 잘 실행이 된다.



여러 가지 조합을 테스트해본 결과 얻어낸 결론:
  1. 런타임 라이브러리를 맞춰줘야 한다. (이건 구글링 결과와 같다)
  2. 그리고 Preprocessor 에서 _DEBUG를 모두 넣어주던가 모두 빼줘야 한다.


2번의 경우 심증이 가는게 있는데, 바로 STL이다. STL의 경우 디버그 모드로 빌드하면 디버그 정보를 담고 경계 검사 등 이것저것 해주는게 많아지는 것으로 알고 있다.

만약 클래스나 구조체가 릴리즈와 디버그 모드일 때 담고 있는 데이터가 달라지게 될 경우, 라이브러리는 이미 코드가 고정되어있는 상태기 때문에 넘겨받은 자료구조 내의 실제로는 존재하지 않는 데이터에 접근한다거나 경계가 달라진다거나 해서 스택이 깨질 수 있다. 이건 실험을 통해 가능하다는 것을 확인했다.

이게 무슨말이냐면, 예를 들어 A라는 라이브러리를 디버그 모드로 빌드했다고 가정하면 이 녀석은 STL 헤더를 디버그용으로 인클루드해서 빌드를 해놓게 된다. 그런데 릴리즈 모드로 B라는 실행 파일이 이 라이브러리를 사용하게 되면 STL 헤더를 릴리즈용으로 인클루드한다. 그러면 쾅~ 서로 자료구조가 같다고 가정하고 링크했지만 실제 자료구조는 다르기 때문에 발생하게 된다. 소켓 통신을 하는데 보내는 사람과 받는 사람이 생각하는 프로토콜 구조가 다른 것과 비슷하다고 할 수 있다.


일단 골치아프지 않으려면 라이브러리도 디버그용과 릴리즈용을 구분해서 쓰는게 좋겠다.



반응형
금요일에 회사에서 Test Driven Development에 대해 소개하는 세미나를 했다.

내가 다니는 회사에서는 한달에 한번씩 프로그래머가 세미나를 진행하는 이벤트(?)가 있다. 저번달 세미나가 끝나고 다음 세미나를 누가 할지에 대해 결정해야 하는데, 보통 서로 잘 하려고 하지 않는다.

그도 그럴 것이 세미나 준비는 상당히 귀찮은 일인데다가 시간도 들고 또 많은 사람들 앞에서 발표를 하는 것이 보통 프로그래머들의 내면 지향적인 성격과는 상당히 맞지 않아서일게다. 약간의 보상은 있긴 하다.

그런데 이번에는 왠지 내가 하고 싶었다. 그 당시에는 - 그러니까 저번 달 초 - '프로그래밍 심리학'이라는 책을 읽고 있던 때였는데, 프로그래밍과 인간의 심리와의 관계를 다룬 고전이었다. 나온지 40년이나 된 책이라서 말은 고전이라지만 그 내용은 지금 봐도 별로 어색한 감이 없었다. 오히려 프로그래머는 '프로그램을 만드는 기계'로 취급되는 지금은 현실은 40년 전에 비해 크게 나이지지 않았다는 사실을 깨닫게 해주었다. 한번 읽어보길 권한다, 정말 재미있다.

그래서 프로그래머의 심리적 요인이 개발에 얼마나 지대한 영향을 미치는지에 대해 강연하기 위해 세미나를 맡겠다고 했다. 그러면 사람들이 좀 더 자신과 남들의 마음을 이해하고, 더 즐거운 마음으로, 그리고 서로에게 상처주지 않으며 개발을 할 수 있게 될 것이라고 생각했기 때문이다.


그러고 나서 며칠 뒤 그러다가 팀원들과 식사를 하다가 세미나 내용에 대해 이야기가 나왔다. 다들 물어보길래 프로그래밍 심리학에 대해 하겠다고 하면서 이런 저런 이야기를 하다가 TDD - Test Driven Development - 를 세미나로 하면 어떨까 하는 생각을 던져봤는데 갑자기 팀원들이 그게 더 낫겠다고 입을 모아 이야기했다. 헉 이런... 프로그래밍 심리학에는 다들 별로 관심이 없는건가?

결국 대중의(?) 의견을 따라 TDD를 세미나 주제로 정했다. 사실 처음에도 프로그래밍 심리학과 TDD 중 어떤 것을 주제로 할까 고민하고 있었던 참에, 남들이 더 관심을 가지는 내용을 하는게 맞겠다고 생각했기 때문이다.


당시에는 나 또한 TDD라는 것에 관심을 가진 지 얼마 되지 않은 시점이었다. TDD 세미나 준비를 하면서 느낀 것은 남들에게 어떤 것을 가르치기 위해서는 나의 이해도와 경험이 청자보다 훨씬 뛰어나야 한다는 점이었다. 어떤 지식을 남에게 전달하는 것은 그 사람에게 내가 알고 있는 것을 이해시키고 경험을 전달하는 것인데, 나조차도 정확하게 경험하지도 이해하지도 못하면 어떻게 가르칠 수 있을까. 따라서 내가 TDD를 해보고 느끼고 배우는 것이 곧 세미나를 성공으로 이끄는 길이라고 생각했다.

처음엔 쉽지 않았다. 시작해보자는 생각은 했지만 마치 C를 처음 배우는 학생처럼 막막하기만 했다. 켄트 백의 '테스트 주도 개발'이라는 책을 읽고, 웹에서 TDD를 경험해본 사람들의 경험담을 뒤적거리기 시작했다. 특히 김창준님의 애자일 이야기 블로그에 포스팅되었던 관련된 글의 경험담이 인상적이었다. 알고 보니 김창준님은 동영상까지 찍으셨다!

박피디님의 세미나 자료도 많은 도움이 되었다.

잡설: 재미있게도 구글에서 '김창준'을 검색할 때와 네이버에서 '김창준'을 검색할 때의 결과가 아주 다르다.


C/C++에서의 TDD는 Java에 환경이 훨씬 열악한 것 같다. 그 중에서는 언어적 특성도 한 몫을 하지만, TDD 관련된 서적과 자료가 대부분 Java를 중심으로 되어 있다는 점이 그것이다. 특히 IDE의 지원도 역시 달랐는데, JUnit + Eclipse 조합으로 볼 수 있는 '빨간 막대''녹색 막대'를 Visual Studio에서는 볼 수 없다는 점이 아주아주아주!! 싫었다(현재는 극복한 상태다). Visual Studio에서는 테스트 결과를 검은 배경에 흰색 글자가 찍힌 콘솔로 봐야 했다.

나중에 찾아낸 GoogleTest콘솔로 찍어줄 때 글자색이 빨간색녹색으로 나온다는 점에서 그런 갈증을 어느 정도 해소해 준다는 점에서 정말 좋았다. 역시 구글이다는 생각이 절로 들었다.

막대 표시해주는 Add-in 찾는 데만 며칠을 썼던 것 같다. 그 중 찾아낸 것이 쑥갓님의 vutpp이다. 이것을 발견한 뒤에야 비로소 첫번째 테스팅을 작성할 수 있었다. 그만큼 Java 유저들이 막대를 보며 느끼는 재미를 나도 느껴보고 싶었던 것이다.


두번째로는 어떤 테스팅 프레임워크를 쓸까 하는 고민을 했다. 이것도 역시 며칠을 소비했다. Noel이라는 사람이 테스팅 프레임워크 6종 비교를 해둔 기사를 보고 고민 끝에 첫 테스트 코드를 CppUnitLite로 작성했다. 가장 쉽고 작으며 내 입맛대로 수정이 가능한 테스팅 프레임워크였다.

그런데 한계가 금방 왔다. 처음에는 몇 가지 테스트를 하면서 재미를 보다가, 테스트가 늘어나자 중복되는 코드가 있어 이것을 묶어주는 Fixture라고 불리는 것이 없었기 때문에 직접 구현해줘야 했다. 또 테스트를 내가 원하는 것만 실행하고 싶을 때가 있는데, 이런 기능이 없었다. 다른 프레임워크에서는 대부분 지원해주기 때문에, 또다시 CppUnitLite 보다 더 나은 테스팅 프레임워크를 찾아 떠돌았다. 그리고 아직 녹색 막대는 포기하지 않은 상황이었다.


그렇게 발견한 것이 Visual Assert였다. Visual Studio와 통합되는 전용 Add-in으로 막대를 볼 수 있었을 뿐만 아니라, Fixture도 지원하고 원하는 테스트만 실행할 수도 있었다! 그렇게 다시 행복을 맛보던 찰나... 한계가 다시 찾아왔다. 이번에는 테스트가 실패해도 다음 테스트를 그냥 진행할 수 있는 기능이 없었던 것이다. 이래가지고는 테스트가 하나만 실패해도 그 뒤쪽에서 기다리는 테스트들은 모두 실행될 수 없게 되어버린다.


결국 마지막으로 찾은 것이 GoogleTest. MS MVP이신 김용현님의 추천이 있었다. 사실 처음에 추천해주셨을 때는 '막대'를 못본다는 점 때문에 별로 신경을 쓰지 않았다. 하지만 상황이 이렇게 되자 다시 기억이 났다. "아, 이걸 한번 써보자!"

써본 결과는 대만족이었다. 딱 하나, Visual Studio에서 막대를 볼 수 없다는 것만 빼면 내가 필요한 기능이 다 들어있었다. 이미 막대는 볼 만큼 봤기 때문에 더 이상 미련을 갖지 않았다. 그리고 텍스트에 색깔이 들어있다는 점에서 어느 정도 만족했다. 이정도면 정말 만들어 준 사람들에게 고맙다는 인사라도 한번 해야 할 것 같다. Thank you!



그리고... 이윽고... 드디어, 세번째 문제가 왔다. 이 문제야 말로 정말 사람프로그래머을 환장하게 하는 문제인데, 바로 Legacy Code라는 놈이다. 현재 진행 중인 프로젝트에 적용하려고 하니 그야말로 시작부터 문제였다. 한발자국도 앞으로 나아갈수가 없었다. 레거시 코드의 종속성은 내 상상을 초월할 정도로 뿌리깊었다. 소스 코드를 몇 시간동안이나 살펴봤지만 대체 어떻게 테스트 코드를 만들어야 할 지 머릿속은 새하얗기만 했다.

단 한개의 메소드만 테스트하려고 해도, 객체를 생성하고 초기화하는 과정에서 제공해줘야 하는 다른 객체가 너무나도 많았다. 환경 설정을 읽어오는 객체는 실제로 환경 설정 파일에 접근하고 있었고, 프로세스 간 통신을 담당하는 객체는 정말로 다른 프로세스를 찾아 연결시켜주는 역할을 했다. 하지만 테스트 과정에서 진짜 환경 설정 파일과 다른 프로세스를 제공해줄 수는 없었기 때문에 테스트 코드 자체가 나올 수 없었다.

물론 Mock Object라는 것을 사용하면 이런 것도 테스트할 수 있다는 점은 이미 알고 있었지만, 문제는 경험이었다. C를 처음 접할 때 포인터라는 것을 배우는 것과 같다. 포인터를 책에서 보고, 정의를 읽고, 사용법을 보고, 예제를 보아도 책을 쓴 사람은 이 정도면 알아들었겠지? 라고 생각했을지 모르나, 배우는 입장에서는 중학생에게 미적분을 보여준 것과 비슷한 얼굴을 하게 될 것이다.

너무나도 막막한 나머지 나는 그야말로 gg를 쳤다. 망했어요~

TDD가 좋다고 알고 있고 아무리 강조해도 무슨 소용인가! 레거시 코드 앞에서 TDD를 도입한다는 것은 쉽지 않다 정도가 아니라 불가능에 가까워 보였다. 주변 사람들에게 TDD라는 걸 써보자고 했을 때 다들 보였던 반응이 그것이었다. 기존 코드에 어떻게 적용할 것이냐고.


일단은 세미나 날짜가 되어서 세미나를 진행했다. 세미나는 TDD의 '실전'이 아니라 '소개'에 중점을 두었다. 이 월례 세미나는 모든 개발자가 의무에 의해 참석하는 것이기 때문에 대부분의 사람들이 TDD에 관심이 없을 가능성이 높다는 것은 알고 있었다. 만약 TDD의 심화 과정을 다뤘다면 - 사실 지금 그정도 내공도 안되지만 - 모두들 세미나 시간에 쓰러져 잤을 것이다. 그래서 전통적인 개발의 단점은 무엇인지, TDD가 어떤 점을 해소해 줄 수 있는지, 어떤 점이 좋은지대 대해 집중적으로 설명하고, 마지막으로 아주 간단한 예제 - 자판기 - 로 시연을 진행했다.

이번 세미나로 최소 한 명 이상이 TDD에 관심을 가지고 첫번째 테스트 코드를 작성해본다면 나는 성공했다고 생각한다.


세미나는 끝났지만, 나의 깊은 고민은 아직 끝나지 않았다. 레거시 코드... 대체 어떻게 해야 할까? 리팩토링에 대한 책도, 패턴에 대한 책도 다시 한번씩 읽어봤지만 별로 도움이 되지 않는다. 이론과 실전이 이렇게나 다를 줄이야.

결국 Working Effectively with Legacy Code, 일명 WELC 원서를 주문했다. 번역판이 있긴 한데 발번역이라고 해서 차라리 원서를 보기로 결정했다. 왠만큼 MSDN을 보고 구글링을 하다 보니 이제 영어도 별로 무섭지 않다.


반응형

'일기' 카테고리의 다른 글

코딩 슬럼프  (0) 2010.11.02
프로그래머와 국회의원의 공통점  (0) 2010.11.02
레거시 코드와 씨름 중  (0) 2010.10.10
프로그래밍 심리학  (0) 2010.09.15
ONE MUST FALL 2097  (0) 2010.02.27
친구가 짠 프로그램 소스를 우연히 보게 되었다.

C로 작성된 - 확장자는 cpp였으나 - 이 프로그램은 친구가 테스트해보라며 나에게 건네줬던 프로그램의 소스였다. 나는 실무 경험이 몇 년 있었지만 그 친구는 대학원생이었기 때문에 실무와는 좀 거리가 있었다. 그래서인지 소스 코드가  학부생 수준의 스타일로 작성되어 있었다. 보자마자 드는 생각이, 어떻게든 뜯어 고치고 싶은 생각뿐이었다. 

그런데 고치는 일이 그리 만만한 일이 아니었다. 일단 작은 프로그램이긴 했지만 멀티 스레드로 동작하고 있었다. 요즘 공부하는 TDD를 적용한다고 치면 - 이 경우엔 기존 코드에 테스트를 씌우는 작업 -  모든 네이티브 함수들을 클래스화해서 Mock object를 사용하도록 바꿔야 했다. 

아... 그런데 몇 줄 안되는 네이티브 C 코드를 수정하는 것도 결코 만만한 일이 아니었다. 이렇게 고통스러울 수가!

그래서 Working Effectively with Legacy Code를 읽기 시작했다. 실무(?)에 앞서 이론을 좀 다져놔야하지 않겠느냐는 생각이었다. 그런데 책이 또 영어라서 이것도 고통스럽다. 이제는 영어를 공부해야 하나...

 
반응형

'일기' 카테고리의 다른 글

프로그래머와 국회의원의 공통점  (0) 2010.11.02
Test Driven Development, 첫 걸음부터 세미나를 하기까지  (0) 2010.10.10
프로그래밍 심리학  (0) 2010.09.15
ONE MUST FALL 2097  (0) 2010.02.27
으아앙대  (0) 2010.02.26

+ Recent posts