메뉴 건너뛰기

창작에 관련된 질문이나 간단한 팁, 예제를 올리는 곳

6화 : 클래스

오늘은 재밌을 수도 있고 어려울 수도 있는 클래스를 만들어 보겠습니다.
일단 지금까지 작업을 보죠.
x, y, z 등 여러가지 변수가 널부러져 있습니다.
그런데 이제 겨우 주인공 하나 만들었을 뿐이거든요.
적1의 변수는 어떻게 할 겁니까?
x2, y2, z2의 변수를 계속 만들어 냅니다.
적2는? x3, y3, z3를 만드는 식으로... 적100까지 필요하다면?

조금 C를 배우신 분들이라면 배열을 써서
x[0], x[1], x[2]식으로 쓰실 수 있을 겁니다.
하지만 이것도 그닥 좋지 못합니다.
x[0]과 y[0]를 묶는 관계라는 것이 보이지 않습니다.
같은 0을 쓰고 있잖아! 라고 반박하신다면 혹시 map[0]과 같은 맵에서 쓰는 배열이 있을 수도 있겠죠.

그깟 관계가 뭔데...
라고 생각하실지 몰라도 중요한 것입니다.
관계가 잘 짜여있으면 코드를 이해하기 쉬워지죠.
아마 지금은 혼자 만들겠지만 다른 사람들이 자기 코드를 쓸 수도 있는 것이고
다른 경우는 자기가 만든 코드는 나중에 잊어버릴 수 있는 겁니다. 흔한 일이죠.

클래스라는 것은 C++의 핵심이라고도 할 수 있을 겁니다.
컴공이라면 객체지향이란 말은 한번씩 들어보셨겠죠.
그 객체지향의 중심에 클래스가 있습니다.

지금까지는 x변수는 주인공의 x좌표야. 그리고 jump_power는 주인공의 점프력이지.
그리고 이 while 루프 안에서 키입력을 받아서 x좌표나 y좌표가 변화하는 거야.
이런 식의 설명을 해야했다면 클래스에서는 더욱 현실에 가까운 설명이 가능합니다.
주인공이란 클래스가 있어. 주인공은 x, y, z 좌표를 가져.
그는 키입력을 받아 행동을 할 수 있어.
...라고 말입니다.

이론적으론 재활용성이나 다형성, 캡슐화등 장점이 있지만
그런건 쓰면서 느끼기로 하고 일단 클래스를 봅시다.

//유닛 클래스
class Unit
{
private:
     int x;                    //X좌표
     int y;                    //Y좌표
     int z;                    //Z좌표점프한높이
     int delay;               //애니메이션용변수
     int state;               //현재상태
     bool left_punch;     //왼팔오른팔펀치질검사
     int jump_power;          //점프력

public:
     void Action();               //상태에따라행동을함
     void Control();               //키입력을받아들임
};

이것이 유닛 클래스입니다.
주인공 클래스가 아닌 이유는 적들도 이것을 쓸 것이기 때문이죠.

class Unit

이 클래스는 Unit이란 이름의 클래스입니다. 간단하죠.
그럼 이 클래스에 대해서 {}안에 정의해봅시다.

변수들은 저번에 설명했던 것들을 다 클래스 안에 넣었을 뿐 별거 없습니다.
늘어난 것은 Action()과 Control()이란 함수 둘이죠. 메소드라고 불립니다.

또 하나 눈 여겨 볼 것은 private와 public으로 나뉘어져 있다는 것.
private: 밑으로 적힌 변수 및 함수들은 프라이버시라는 거죠.
침범하지 말아 달라는 겁니다. 이건 자기만 알고 자기만 쓸 것이라는 것.
그럼 public:은 무엇이냐? 맘대로 쓰라는 거죠.
그 의미는 나중에 쓰면서 더 자세히 알기로 합시다.

다음은 변수를 초기화 해봅시다.
지금은 x=0 식으로 초기화가 안 되어 있습니다.
만약 x=0을 저기에 적으면 에러가 납니다.
클래스에서는 초기화해야 할 부분은 따로 있습니다.

public 부분에 Unit()이라고 추가하고(앞에 int, void 같은 건 안 적습니다.)
밑에 Unit()에 대해 적어줍니다.

Unit::Unit()

     x = 100; 
     y = 100; 
     z = 0; 
     delay = 0; 
     state = 0; 
     left_punch = false;
}

클래스의 메소드는

(리턴 형태) 클래스명::메소드명()
{
}

식으로 보통 선언을 합니다.
그런데 이것은 특이하게 리턴형태가 없는데다가
클래스명=메소드명입니다.
이것을 생성자라고 부릅니다.
즉, 이 클래스가 생겨날 때 실행되는 메소드입니다.
그 정확한 타이밍은 나중에 코드를 쓰며 가르쳐 드리겠습니다.
일단 생성자란 초기화 내용을 적는 것으로 초기화 내용은 전의 그것과 똑같습니다.

그럼 나머지 Action, Control에 대해서도 정의해 봅시다.

void Unit::Control()
{
     if(GetKey(vkey_up,1))
     {
          y-=5;
     }

     if(GetKey(vkey_down,1))
     {
          y+=5;
     }

     if(GetKey(vkey_left,1))
     {
          x-=5;
     }

     if(GetKey(vkey_right,1))
     {
          x+=5;
     }

     if(GetKey(vkey_z))
     {
          state=1;
          delay=0;

          if( left_punch)left_punch=false;
               else left_punch=true;
     }

     if(GetKey(vkey_x))
     {
          state=2;
          delay=0;
     }

     if(GetKey(vkey_c))
     {
          state=3;
          jump_power=100;
     }

     if(x<0)x=0;
     if(x>SCREEN_X-38)x=SCREEN_X-38;
     if(y<125)y=125;
     if(y>SCREEN_Y-58)y=SCREEN_Y-58;
}

void Unit::Action()
{
     //평상시
     if(state==0)
     {
          if(delay<30)jdd->DrawPicture(backbuffer, "걷기2", x, y-2, NULL);
               else if(delay<60)jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL);
               else if(delay<90)jdd->DrawPicture(backbuffer, "걷기3", x, y, NULL);
               else jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL);

               ++delay;
               if(delay >= 120)delay=0;
     }

     //펀치
     else if(state==1)
     {
          if(left_punch)jdd->DrawPicture(backbuffer, "펀치1", x-8, y-2, NULL);
               else jdd->DrawPicture(backbuffer, "펀치2", x-8, y-2, NULL);

          ++delay;
          if(delay>=10)state=0;
     }

     //킥
     else if(state==2)
     {
          jdd->DrawPicture(backbuffer, "킥", x-6, y, NULL);

          ++delay;
          if(delay>=10)state=0;
     }

     //점프
     else if(state==3)
     {
          jdd->DrawPicture(backbuffer, "걷기1", x, y-z, NULL);

          z+=jump_power/10;
          jump_power-=gravity;
          
          //착지
          if(z <= 0)
          {
               z=0;
               state=0;
          }
     }
}

뭔가 잔뜩 적었지만 보면 이전에 적었던 코드 그대로 잘라내기/붙여넣기 한 겁니다.
Control()에는 키 입력 관련 부분을, 그리고 Action 부분에는 행동 관련 부분을.
그런데 이대로라면 gravity 때문에 에러가 날 겁니다.
왜냐면 gravity는 main에 선언되어 있기 때문에 여기서 사용할 수 없기 때문입니다.
그렇다고 캐릭터당 중력을 따로 가지는 것은 말이 안 되므로 class 안에 넣기도 그렇습니다.
그럼 방법은 2가지입니다. 상수로 만들어 버리든지 전역변수로 해버리든지.
저희는 일단 9를 고정 값으로 할 테니 상수로 만들어 버리겠습니다.
참고로 상수는 변수와 다르게 '항상 그 숫자'라고 생각하시면 될 겁니다.

윗부분의 #include 밑에 쯤에

#define GRAVITY      9

라고 적어줍시다.
그리고 jump_power-=gravity; 부분을
jump_power-=GRAVITY; 로 고쳐줍시다.

상수는 꼭 영대문자로 해야할 필요는 없지만 하나의 룰이니 지켜주는 게 좋습니다.
이 상수와 전처리의 유용성에 대해서는 나중에 다시 설명할 기회가 있을 겁니다.

그럼 최종적으로 코드는 이런 모습이 되어 있을 겁니다.
만약 도중에 '이걸 도대체 어디 넣어? 라고 생각하셨던 분이라면 이 코드를 보고 수정하십시오.

#include <stdio.h>
#include <math.h>

#include "donglib.h"
#include "dscript.h"

#define GRAVITY     9

bool window_mode=true;

//유닛 클래스
class Unit
{
private: 
     int x;                    //X좌표 
     int y;                    //Y좌표 
     int z;                    //Z좌표(점프한 높이) 
     int delay;               //애니메이션용 변수 
     int state;               //현재 상태 
     bool left_punch;     //왼팔, 오른팔 펀치질 검사 
     int jump_power;          //점프력

public: 
     void Action();               //상태에 따라 행동을 함 
     void Control();               //키 입력을 받아 들임 

     Unit();
};

Unit::Unit()

     x = 100; 
     y = 100; 
     z = 0; 
     delay = 0; 
     state = 0; 
     left_punch = false;
}

void Unit::Control()

     if(GetKey(vkey_up,1)) 
     { 
          y-=5; 
     } 

     if(GetKey(vkey_down,1)) 
     { 
          y+=5; 
     } 

     if(GetKey(vkey_left,1)) 
     { 
          x-=5; 
     } 

     if(GetKey(vkey_right,1)) 
     { 
          x+=5; 
     } 

     if(GetKey(vkey_z)) 
     { 
          state=1; 
          delay=0; 

          if( left_punch)left_punch=false; 
               else left_punch=true; 
     } 

     if(GetKey(vkey_x)) 
     { 
          state=2; 
          delay=0; 
     } 

     if(GetKey(vkey_c)) 
     { 
          state=3; 
          jump_power=100; 
     } 

     if(x<0)x=0; 
     if(x>SCREEN_X-38)x=SCREEN_X-38; 
     if(y<125)y=125; 
     if(y>SCREEN_Y-58)y=SCREEN_Y-58;
}

void Unit::Action()

     //평상시 
     if(state==0) 
     { 
          if(delay<30)jdd->DrawPicture(backbuffer, "걷기2", x, y-2, NULL); 
               else if(delay<60)jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL); 
               else if(delay<90)jdd->DrawPicture(backbuffer, "걷기3", x, y, NULL); 
               else jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL); 

               ++delay; 
               if(delay >= 120)delay=0; 
     } 
     //펀치 
     else if(state==1) 
     { 
          if(left_punch)jdd->DrawPicture(backbuffer, "펀치1", x-8, y-2, NULL); 
               else jdd->DrawPicture(backbuffer, "펀치2", x-8, y-2, NULL); 

          ++delay; 
          if(delay>=10)state=0; 
     } 
     //킥 
     else if(state==2) 
     { 
          jdd->DrawPicture(backbuffer, "킥", x-6, y, NULL); 

          ++delay; 
          if(delay>=10)state=0; 
     } 
     //점프 
     else if(state==3) 
     { 
          jdd->DrawPicture(backbuffer, "걷기1", x, y-z, NULL); 
          z+=jump_power/10; 
          jump_power-=GRAVITY; 

          //착지 
          if(z <= 0) 
          { 
               z=0; 
               state=0; 
          } 
     }
}

//////////////////////////////////////////////////
//메인 코드
//////////////////////////////////////////////////
int main(char* arg[])

     //창 생성 
     if(!MainInitialize("Sample", TRUE, FALSE, window_mode))return 0; 

     //윈도우창 이동 
     if(window_mode) 
     { 
          jdd->OnMove(100, 100); 
          SetCursor(LoadCursor(0, IDC_ARROW)); 
     } 

     JPictureInfo jpi; 
     jpi.SetColorKey(JColor(0,0,255)); 

     jdd->LoadPicture("배경", "던젼6.jpg", NULL, true); 
     jdd->LoadPicture("걷기1", "w1.gif", &jpi, true); 
     jdd->LoadPicture("걷기2", "w2.gif", &jpi, true); 
     jdd->LoadPicture("걷기3", "w3.gif", &jpi, true); 
     jdd->LoadPicture("펀치1", "p1.gif", &jpi, true); 
     jdd->LoadPicture("펀치2", "p2.gif", &jpi, true); 
     jdd->LoadPicture("킥", "k1.gif", &jpi, true); 

     //메인 실행 
     while(!GetKey(vkey_esc)) 
     { 
          if(!ProcessMessage())break; 
          jdd->DrawPicture(backbuffer, "배경", 0, 0, NULL); 
          jdd->Render(); 
     } 

     //정리하고 끝내기 
     jdd->DeleteFont(global_font); 

     return 0;
}

//윈도우 메세지 처리
LRESULT CALLBACK WndProc(HWND wnd,UINT msg,WPARAM wParam,LPARAM lParam)

    switch ( msg ) 
    { 
          case WM_LBUTTONDOWN  :     if(mouse_control)LButton=TRUE; 
                                                           SetCapture(hwnd); 
                                                           break; 

          case WM_RBUTTONDOWN  :     if(mouse_control)RButton=TRUE; 
                                                           SetCapture(hwnd); 
                                                           break; 

          case WM_LBUTTONUP    :     if(mouse_control)LButton=FALSE; 
                                                        ReleaseCapture(); 
                                                        break; 

          case WM_RBUTTONUP    :     if(mouse_control)RButton=FALSE; 
                                                        ReleaseCapture(); 
                                                        break; 

          case WM_MOUSEMOVE    :     if(mouse_control) 
                                                        { 
                                                               MouseX=LOWORD(lParam); 
                                                               MouseY=HIWORD(lParam); 
                                                               mouse_move = TRUE; 
                                                               ShowCursor(FALSE); 
                                                       } 
                                                       break; 

          case MM_MCINOTIFY    :     /*if ( ReplayFlag && wParam == MCI_NOTIFY_SUCCESSFUL ) */_MidiReplay(); 
                                                     break; 

          case WM_DESTROY       :     _CrtDumpMemoryLeaks(); 
                                                      break; 

          case WM_CLOSE           :     gameover=TRUE; 
                                                       break; 

          case WM_SIZE           :     if(wParam == SIZE_MINIMIZED)activate=false; 
                                                 else activate=true; 
                                                 break; 

          case WM_MOVE           :     if(jdd)jdd->OnMove(LOWORD(lParam), HIWORD(lParam)); 
                                                     break; 

          case WM_ACTIVATE      : if(LOWORD(wParam))activate=true; 
                                                  else activate=false; 
                                                  break; 
     } 

     return DefWindowProc(wnd,msg,wParam,lParam);
}

실행시켜 보면 배경만 나옵니다.
그럴 수 밖에요. 아직 클래스만 만들어 놓고 선언을 안 했으니까요.
변수 선언하듯 캐릭터를 선언해 봅시다.
물론 while 루프에 들어가기 전에 선언해야 겠죠.

Unit hero;

이러면 마치 int나 bool처럼 Unit 형태의 hero가 생기는 겁니다.
하지만 이건 변수가 아니고 인스턴스라고 불립니다. 이론적인 것은 접어두고.

앞에 선언했던 Unit::Unit()은 바로 이 선언부분에서 실행됩니다.
바로 hero를 Unit형으로 선언함과 동시에 x=100, y=100 등의 값이  할당되는 겁니다.

이제 while문 안에 다음 두 코드를 추가합니다.

hero.Control();
hero.Action();

끝입니다. 기능은 저번과 똑같습니다.
뭐가 저번에 비해서 달라졌는지 아직 모르겠다는 분들이 있으시겠지만
게임의 규모가 커질 수록 그 강점이 눈에 보일 겁니다.

내일은 현재 공중에서 이단 점프를 하거나 주먹 내밀다가 이동하는 등의
버그같은 주인공의 동작을 다듬어 보겠습니다.

 

따로 오시지 못하는 분들은 위한 소스.
이걸 그대로 붙여 쓰세요.
main.cpp

조회 수 :
3114
등록일 :
2008.04.23
23:21:58 (*.239.144.2)
엮인글 :
게시글 주소 :
https://hondoom.com/zbxe/index.php?mid=study&document_srl=192203

A.미스릴

2008.04.25
04:36:48
(*.234.10.203)

아무래도 클레스 강의를 위해서는 함수강의가 먼저 필요함

똥똥배

2008.04.25
05:01:06
(*.193.78.73)
전 이론보다는 실용파라서 함수가 뭔지 몰라도 저렇게 써보고 이해하면 된다고 봅니다.
안 되면 설명해줘야 겠죠.
저같은 경우 처음부터 이론 없이 남의 코드를 보고 프로그래밍을 배운 케이스라서...
List of Articles
번호 제목 글쓴이 조회 수 추천 수 날짜 최근 수정일sort
105 문D라이브로 더블드래곤을 만들자(3) file 똥똥배 3430   2008-04-18 2008-04-18 18:27
 
104 문D라이브로 더블드래곤을 만들자(4) [2] file 똥똥배 3213   2008-04-20 2008-04-21 07:20
 
103 C++ 데이터의 바이트 용량 임의로 정의할수 없나영 [1] A.미스릴 3059   2008-04-21 2008-04-21 07:18
int는 4바이트로 정해져 있는데 약간의 수만 있으면 되는 수도 있는데 괜히 많은 숫자를 사용해서 메모리를 많이 사용하는 건 아닐지 ㅡㅡ; 3바이트라던지 4비트라던지... 데이터의 바이트 사용량을 임의로 바꿀수 없나요  
102 문D라이브로 더블드래곤을 만들자(5) [6] file 똥똥배 3072   2008-04-21 2008-04-24 03:23
 
» 문D라이브로 더블드래곤을 만들자(6) [2] file 똥똥배 3114   2008-04-23 2008-04-25 05:01
6화 : 클래스 오늘은 재밌을 수도 있고 어려울 수도 있는 클래스를 만들어 보겠습니다. 일단 지금까지 작업을 보죠. x, y, z 등 여러가지 변수가 널부러져 있습니다. 그런데 이제 겨우 주인공 하나 만들었을 뿐이거든요. 적1의 변수는 어떻게 할 겁니까? x2, y...  
100 문D라이브로 더블드래곤을 만들자(7) file 똥똥배 3480   2008-04-27 2008-04-28 04:22
 
99 문D 라이브 질문 [5] 대슬 2889   2008-05-15 2008-05-16 06:13
1. 음악이나 사운드는 어떻게 불러와서 출력하나요. 배웠었는데 까먹었음. 2. 그림을 원하는 각도로 자유롭게 회전시켜서 출력하는 기능은 없나요?  
98 문D라이브로 더블드래곤을 만들자(8) [1] file 똥똥배 2973   2008-05-16 2009-01-07 22:05
 
97 이상하군요. [2] 대슬 2490   2008-05-16 2009-01-07 22:05
MIDI 음악을 메인 루프 앞에서 재생시킨 후 다른 음악을 재생시키려고 했더니 그 다음부턴 음악이 정지만 하고 재생이 되지를 않네요. 예전에 배고파요 만들었을 때는 비슷하게 해도 별 문제가 없었던 것 같은데, 왜 이러는지 잘 모르겠습니다. void bgm(int ...  
96 문D라이브로 더블드래곤을 만들자(9) 똥똥배 2889   2008-05-17 2008-05-17 03:07
9편 : 체계화된 동작 지금까지 열심히 이 강의를 따라오신 분들이라면 지금 게임의 여러 버그가 산재해 있음을 깨달으셨을겁니다. 그걸 왜 그냥 내버려 뒀나면... 귀찮아서~ ~는 훼이크고 일단 가르치는 주제에서 벗어나면 집중력이 떨어지고 어느 정도 이해력...  
95 문D라이브로 더블드래곤을 만들자(10) file 똥똥배 4065   2008-05-17 2008-05-17 03:07
 
94 문D라이브로 더블드래곤을 만들자(11) file 똥똥배 4647   2008-05-17 2008-05-17 20:24
 
93 #define에 대해 [1] A.미스릴 4061   2008-05-19 2008-05-19 18:57
#define을 너무 많이 쓰면 코드에 불안정화가 온다는데 사실인가여? 그러면 불안정화를 막으러면 어떻게 해야하나여 아니면 enum이나 const를 써야 하나...  
92 문D 질문 [5] A.미스릴 3847   2008-05-26 2008-06-12 18:55
1.글자, 문장을 화면상에 출력하는 방법 2.숫자를 스트링으로 형변환할떄 아스키 코드가 아니라 숫자가 옮겨가게 하는법 3.사각형, 원 등을 출력하는 방법(Rectangle 함수같은거 없나여)  
91 문D 질문 #2 [1] A.미스릴 8797   2008-06-01 2009-01-07 22:05
이거 도배하는거같은 마음이 드네여 ㅈㅅ 제가 대화창을 다뤄봤는데여 CTextDlg textdialog(50); CTextDlg * pt_textdialog(&textdialog); pt_textdialog->SetDlg(100, 100, 200, 200); pt_textdialog->SetColor(255,255,0); pt_textdialog->SetDlgBox("_d...  
90 문D라이브 2008년 5월 19일 버전 [2] 혼돈 3747   2008-06-02 2015-12-09 08:06
휴가루 웍휴2를 개발하면서 조금씩 수정 중이지만, 미스릴님의 요청이 있어서 올립니다. 대화창에 관련되서 완성되었다는 것과 흥크립트 부분이 많이 변했다는 것이 주요 변경점입니다. 대화창의 경우 기본은 대화창이 없는 상태로 MakeDlgBox를 사용하면 크기...  
89 MFC 더블 버퍼링 질문 [2] A.미스릴 6602   2008-06-13 2013-11-23 08:43
void CPingpongView::OnDraw(CDC* pDC) { CPingpongDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here // 현재 창 크기와 같은 DC를 만듬 CRect rect; GetClientRect(&rect); int int_client_width = (rect.ri...  
88 씨언어 질문 (내일 시험 ㄷㄷ) [1] 쿠로쇼우 4331   2008-06-17 2008-06-17 22:10
1 121 12321 1234321 123454321 부탁 ㄷㄷ  
87 클레스들을 담은 헤더들의 혼란 [4] A.미스릴 3892   2008-06-21 2013-11-23 08:43
xxx.h파일 //xxx.h class xxx { ... 어떤 멤버함수 (yyy 뭐시기); ... } yyy.h파일 //yyy.h class yyy { ... 어떤 멤버함수 (xxx 뭐시기); ... } 이 두 클래스의 헤더 파일이 있으면 #include "xxx.h"와 #include "yyy.h"를 어떤 순서로 배치해야 할지 혼란스...  
86 문D라이브도 더블버퍼링이 필요한가요? [3] A.미스릴 4223   2008-06-28 2008-06-29 07:06
보니까 backbuffer에 그림을 붙이던데 backbuffer가 배경 버퍼니까 바로 화면에 출력되는건가요? 그런데 그걸 직접 쓰면 MFC의 DC처럼 깜박임과 속도저하가 오는건 아니죠?ㅡ,.ㅡ; 추가로 LoadPicture로 불러온 그림을 다시 안불러온것처럼 되돌리는건 없나요?