메뉴 건너뛰기

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

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
번호 제목 글쓴이 조회 수 추천 수 날짜 최근 수정일
» 문D라이브로 더블드래곤을 만들자(6) [2] file 똥똥배 3114   2008-04-23 2008-04-25 05:01
6화 : 클래스 오늘은 재밌을 수도 있고 어려울 수도 있는 클래스를 만들어 보겠습니다. 일단 지금까지 작업을 보죠. x, y, z 등 여러가지 변수가 널부러져 있습니다. 그런데 이제 겨우 주인공 하나 만들었을 뿐이거든요. 적1의 변수는 어떻게 할 겁니까? x2, y...  
84 문D라이브로 더블드래곤을 만들자(5) [6] file 똥똥배 3072   2008-04-21 2008-04-24 03:23
 
83 C++ 데이터의 바이트 용량 임의로 정의할수 없나영 [1] A.미스릴 3059   2008-04-21 2008-04-21 07:18
int는 4바이트로 정해져 있는데 약간의 수만 있으면 되는 수도 있는데 괜히 많은 숫자를 사용해서 메모리를 많이 사용하는 건 아닐지 ㅡㅡ; 3바이트라던지 4비트라던지... 데이터의 바이트 사용량을 임의로 바꿀수 없나요  
82 문D라이브로 더블드래곤을 만들자(4) [2] file 똥똥배 3213   2008-04-20 2008-04-21 07:20
 
81 문D라이브로 더블드래곤을 만들자(3) file 똥똥배 3430   2008-04-18 2008-04-18 18:27
 
80 문D라이브로 더블드래곤을 만들자(2) [6] file 똥똥배 3611   2008-04-18 2013-11-23 08:43
 
79 VC++ 2008 Express Edition에서 문D라이브 링크 [2] A.미스릴 13090   2008-04-17 2014-04-15 17:28
SDK도 최신버전으로 깔아주었고 도구 설정도 적절하게 했습니다 모든 셋팅이 완료되고 링크를 해보니까 아래와 같은 문장이 쭉쭉나오는군요 warning은 굉장히 많이 나오는데 실제로 컴파일을 막아버린 부분은 LINK : fatal error LNK1104: 'LIBC.lib' 파일을 ...  
78 문D라이브로 더블드래곤을 만들자(1) [2] file 똥똥배 4479   2008-04-16 2008-04-17 08:29
 
77 임의의 점이 다각형 내부에 있는지 검사하는 함수 똥똥배 5358   2008-04-14 2008-04-14 02:21
수학도 기하학도 허접한 제가 짠 거라서 허접하지만 넓은 아량으로 봐 주십시오. struct Point { double x; double y; }; bool LineIn(Point p1, Point p2, Point p3) { double top, bottom; //위 아래 범위 결정 if(p1.y < p2.y) { top = p1.y; bottom = p2.y...  
76 명령문 질문드립니다. [1] X-tra 2244   2008-03-26 2008-03-27 00:40
변해서 게시판 찾기 어려울것 같았는데 이상하게 한번에 찾아 버렸네요. 연구소라고 적혀 있어서요... 이번건 질문이라고 하기에는 좀 애매합니다. 예전에 흥크립트의 모든 명령어가 적혀 있던 파일을 본 기억이 납니다. 하지만 아무리 찾아도 없더군요. 블로...  
75 저는 사실 이걸 잘 못해요. [3] 앟랄 2282   2008-03-26 2008-04-18 07:45
알만툴 액알. 원리가 도저히 이해가 안 갑니다. 그림넣어서 설명해주세요. 그 뭐시기 그냥 근접공격만 일단. 나중가서 원거리를 합시다.  
74 OgreOde 사용기 똥똥배 3184   2008-03-25 2008-03-25 21:37
OgreSDK 버전 1.4.7 OgreOde 버전 0.95(아마도) 사실 오우거 엔진 쓴지도 얼마 안 되고 물리엔진은 처음 만져봤습니다. 처음에 Ogre Wiki에 나온대로 따라서 만들었는데 crateCube.mesh와 plane.mesh 때문에 에러가 나서 바닥은 직접 만들고 crateCube 대신 오...  
73 웹 프로그래밍을 배우려고 합니다. [2] Kadalin 2171   2008-03-22 2008-03-22 23:37
사용하는 형식에 따라 꽤 분야가 많은 것으로 알고 있습니다. 각각의 특징을 설명해 주세요.  
72 질문요.C++ [7] 질문자01 2729   2007-05-10 2008-04-03 05:32
도대체 파라미터가 뭔지 모르겠어요.  
71 흥크립트 클릭명령 질문입니다. [1] 카시 2009   2008-03-18 2008-03-18 20:18
으음 제가 몇가지 실험해본 결과. 1. @영역선택과 @클릭 명령은 한 스크립트 내에서 혼합해서 쓸 수 없다. 2. @영역선택은 ~영역선택으로 쓸 수 없다. (대화창과 동시 처리 불가능) 3. @클릭명령 사용시 오른쪽 버튼을 누르면 세이브화면이 뜬다. 이 세가진데...  
70 흥크립트 키입력 질문 [1] A.미스릴 2363   2008-03-15 2008-03-17 04:37
키 클릭을 누르고 누른상태로 있는것도 인식할 수 있나영? 어떻게 하나요 ㅡ,.ㅡ  
69 그림 출력의 순서를 알고 싶습니다. [3] X-tra 2466   2008-03-12 2008-03-17 04:37
제 글이 연구소로 가서 이번에는 신경써서 제목을 적었습니다 ㅡ.ㅡb 채색을 제외하면 전투도 70%정도 완성되었는데 자잘한 버그가 발생해서요. 제 스스로 알아 내려 했지만 확실하게 하기 위해서 물어 봅니다. 정확히 어떤 그림이 앞에 출력이 되고 어떤 그...  
68 '@클릭'이거 어떻게 사용하는거죠? [4] 네모상자 2244   2008-01-26 2008-03-17 04:37
'@영역선택'사용법은 역전심판 뜯어봐서 알 수 있겠는데, 이건 던전 앤 러버를 뜯을 수도 없고 해서 잘 모르겠네요. * 똥똥배님에 의해서 게시물 이동되었습니다 (2008-03-11 17:11)  
67 퀴즈소스입니다 허클베리핀님 라컨 2546   2005-08-23 2008-03-17 04:37
Private Sub Form_Click() If 시작 <= 1 Then MsgBox ("이제 왔군 학생") MsgBox ("이번이 시험인건 알고 있겠지?") MsgBox ("문제를 낼테니 잘 풀어라") 문제 = InputBox("흑곰의 이전 닉네임은 무엇이였나?") If 문제 = "답" Then MsgBox ("맞았군") Cls Pri...  
66 게임 만들때는 게임 기획을 해야 합니다. [7] 똥똥배 2106   2008-02-12 2008-03-17 04:37
당연한 소리지만 간혹 이것을 잘못하는 사람도 있고 저 역시 이런 실수를 많이 저질렀습니다. 마왕이 용사를 무찌르는 게임을 만들자. 이건 게임 기획이 아닙니다. 시나리오 기획입니다. 이 결과 게임이 어떻게 만들어지는지 봅시다. 일단 마왕이 용사를 무찌...