메뉴 건너뛰기

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

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 C++ 질문 [1] A.미스릴 3818   2008-12-21 2008-12-21 19:24
str_list1[loop_v]가 존재하면 true, 안 존재하면 false 이런 감별이 필요한데 CString str_list1[] = {"","","웹디자인","3D그래픽스"}; int loop_v = 0; while(str_list1[loop_v]) { this->m_ctrl_combobox1.AddString(str_list1[loop_v]); this->m_ctrl_lis...  
104 srand에 관해서 [4] A.미스릴 3868   2008-07-15 2008-07-18 05:49
제가 알고 있는 난수의 사용방법은 srand( (unsigned int) time( NULL ) ); 로 srand를 지정하고 rand()를 해서 랜덤난수를 가져오는 것인데 함수값이 어쩌고 하는 원리더군요 그런데 원리를 생각해 보니 이게 정말 난수의 역할을 해낼수 있을까요?ㅡ,.ㅡ; 애...  
103 COgg 질문 [3] A.미스릴 3710   2008-06-29 2013-11-23 08:43
밑에 대슬님 질문 보니까 멤버변수를 등록할때 동적할당을 하는게 씌여 있는데 그냥 Cogg m_ogg_oggplayer; 이런식으로 그냥 일반변수로 등록을 하고 사용할순 없나요? 동적할당을 굳이 써야하는지 ㅡ.ㅡ;; (개인적으로 4바이트 추가와 잘못되면 메모리 누수가...  
102 문D라이브도 더블버퍼링이 필요한가요? [3] A.미스릴 4223   2008-06-28 2008-06-29 07:06
보니까 backbuffer에 그림을 붙이던데 backbuffer가 배경 버퍼니까 바로 화면에 출력되는건가요? 그런데 그걸 직접 쓰면 MFC의 DC처럼 깜박임과 속도저하가 오는건 아니죠?ㅡ,.ㅡ; 추가로 LoadPicture로 불러온 그림을 다시 안불러온것처럼 되돌리는건 없나요?  
101 클레스들을 담은 헤더들의 혼란 [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"를 어떤 순서로 배치해야 할지 혼란스...  
100 씨언어 질문 (내일 시험 ㄷㄷ) [1] 쿠로쇼우 4331   2008-06-17 2008-06-17 22:10
1 121 12321 1234321 123454321 부탁 ㄷㄷ  
99 문D라이브 2008년 5월 19일 버전 [2] 혼돈 3747   2008-06-02 2015-12-09 08:06
휴가루 웍휴2를 개발하면서 조금씩 수정 중이지만, 미스릴님의 요청이 있어서 올립니다. 대화창에 관련되서 완성되었다는 것과 흥크립트 부분이 많이 변했다는 것이 주요 변경점입니다. 대화창의 경우 기본은 대화창이 없는 상태로 MakeDlgBox를 사용하면 크기...  
98 문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...  
97 문D 질문 [5] A.미스릴 3847   2008-05-26 2008-06-12 18:55
1.글자, 문장을 화면상에 출력하는 방법 2.숫자를 스트링으로 형변환할떄 아스키 코드가 아니라 숫자가 옮겨가게 하는법 3.사각형, 원 등을 출력하는 방법(Rectangle 함수같은거 없나여)  
96 #define에 대해 [1] A.미스릴 4061   2008-05-19 2008-05-19 18:57
#define을 너무 많이 쓰면 코드에 불안정화가 온다는데 사실인가여? 그러면 불안정화를 막으러면 어떻게 해야하나여 아니면 enum이나 const를 써야 하나...  
95 문D라이브로 더블드래곤을 만들자(11) file 똥똥배 4647   2008-05-17 2008-05-17 20:24
 
94 문D라이브로 더블드래곤을 만들자(7) file 똥똥배 3480   2008-04-27 2008-04-28 04:22
 
» 문D라이브로 더블드래곤을 만들자(6) [2] file 똥똥배 3114   2008-04-23 2008-04-25 05:01
6화 : 클래스 오늘은 재밌을 수도 있고 어려울 수도 있는 클래스를 만들어 보겠습니다. 일단 지금까지 작업을 보죠. x, y, z 등 여러가지 변수가 널부러져 있습니다. 그런데 이제 겨우 주인공 하나 만들었을 뿐이거든요. 적1의 변수는 어떻게 할 겁니까? x2, y...  
92 문D라이브로 더블드래곤을 만들자(5) [6] file 똥똥배 3072   2008-04-21 2008-04-24 03:23
 
91 C++ 데이터의 바이트 용량 임의로 정의할수 없나영 [1] A.미스릴 3059   2008-04-21 2008-04-21 07:18
int는 4바이트로 정해져 있는데 약간의 수만 있으면 되는 수도 있는데 괜히 많은 숫자를 사용해서 메모리를 많이 사용하는 건 아닐지 ㅡㅡ; 3바이트라던지 4비트라던지... 데이터의 바이트 사용량을 임의로 바꿀수 없나요  
90 문D라이브로 더블드래곤을 만들자(4) [2] file 똥똥배 3213   2008-04-20 2008-04-21 07:20
 
89 문D라이브로 더블드래곤을 만들자(3) file 똥똥배 3430   2008-04-18 2008-04-18 18:27
 
88 문D라이브로 더블드래곤을 만들자(2) [6] file 똥똥배 3611   2008-04-18 2013-11-23 08:43
 
87 VC++ 2008 Express Edition에서 문D라이브 링크 [2] A.미스릴 13090   2008-04-17 2014-04-15 17:28
SDK도 최신버전으로 깔아주었고 도구 설정도 적절하게 했습니다 모든 셋팅이 완료되고 링크를 해보니까 아래와 같은 문장이 쭉쭉나오는군요 warning은 굉장히 많이 나오는데 실제로 컴파일을 막아버린 부분은 LINK : fatal error LNK1104: 'LIBC.lib' 파일을 ...  
86 문D라이브로 더블드래곤을 만들자(1) [2] file 똥똥배 4479   2008-04-16 2008-04-17 08:29