메뉴 건너뛰기

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

7편 : 적 만들기

원래는 주인공의 동작을 다듬으려고 했었는데
계속 주인공만 만지고 있으니 재미없고 실력되시는 분들은 알아서 고쳤을 지도 모르고
무엇보다도 왠지 동작 설명하고 싶은 기분이 아니라서
적만들기를 먼저 이야기하겠습니다.
저번에 클래스를 만들어 둔 덕분에 손쉽게 적을 만들 수 있습니다.

     Unit hero;
     Unit enemy;

     //메인 실행
     while(!GetKey(vkey_esc))
     {
          if(!ProcessMessage())break;

          jdd->DrawPicture(backbuffer, "배경", 0, 0, NULL);

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

          jdd->Render();
     }

단 두줄을 입력하는 것으로 캐릭터가 하나 더 출현하는 것을 볼 수 있습니다.



주인공과 똑같이 생겼다는 점은 일단 접어두고요.
영역을 벗어나서 벽에 붙어 있습니다.
물론 enemy.control()을 붙이면 정상적으로 되겠지만
적은 조종할 것이 아니므로 이건 아니겠죠.

한마디로 실수한 겁니다. 영역을 제한하는 부분을
Control() 부분에서 Action()의 앞부분으로 옮겨 줍시다.

void Unit::Action()
{
     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;
     ...

처음에는 둘이 같은 위치에 나오므로 '에러 아닌가?' 생각할 텐데
방향키로 움직여 보면 두 명이라는 것을 알 수 있을 겁니다.
그럼 이런 헷갈림을 없애고 제대로 된 게임을 만들기 위해서
초기 위치를 정할 수 있게 하겠습니다.
지금 초기 위치는 어디서 정하고 있죠?
바로 생성자에서 x=100, y=100으로 억지로 정하고 있습니다.
이것을 파라미터를 넣어서 임의로 정할 수 있게 합시다.
일단 클래스 정의 부분에서 단순히 Unit(); 으로 선언되어 있는 것을
Unit(int x, int y) 와 같이 바꿉니다.
이걸 하지 않으면 오버로드 어쩌구 저쩌구 에러가 납니다.
선언부분을 고쳤으니 정의 부분도 고쳐봅시다.

Unit::Unit(int x, int y)
{
     this->x = x;
     this->y = y;
     z = 0;
     delay = 0;
     state = 0;
     left_punch = false
}

this->x = x 부분이 조금 난해할 수 있을 텐데,
설명을 하자면 현재 파라미터로 선언된 int x, int y가 있습니다.
그리고 클래스 내부에 선언되어 있는 int x, int y가 있습니다.
두 개의 이름이 똑같아서 구분하기 위해서
내부에 있는 x는 this->x, 파라미터로 선언된 x는 그냥 x로 쓰는 것입니다.
왜냐면 저 때 지역변수로 선언한 파라미터 변수가 우선순위가 높기 때문입니다.

저게 골아프면 이렇게 써도 되긴 합니다.

Unit::Unit(int vx, int vy)
{
     x = vx;
     y = vy;
    ...

이 경우는 변수 이름이 겹치지 않기 때문에 그냥 원래 이름으로 적어도 됩니다.
하지만 vx가 x와 같은 의미라는 점에서 헷갈리고 저런식으로 변수를 정하다보면
나중에 더 복잡해 지므로 this->x = x 식으로 정하는 것이 보기 좋습니다.
JAVA에서는 이게 정석이기도 하고요. 지금은 VC++지만...

아직 실행하면 안 됩니다.
마지막 할 일이 남았습니다.
자, 저번에 배웠던 것을 다시 생각해 봅시다.
생성자는 어느 시점에서 실행된다고 했죠?

Unit hero;

바로 이 시점입니다. 그러므로 저 파라미터를 넣어 주는 것도 여기서 해야 합니다.

 Unit hero(300, 200);
 Unit enemy(100, 300);

이렇게 hero와 enemy의 선언부를 고쳐봅시다.
그럼 서로 다른 위치에서 시작하는 두 유닛을 볼 수 있을 겁니다.



이젠 적이 움직이게 합시다.
일단 Control과 또 다른 AI라는 메소드를 만듭니다.

 

//유닛 클래스

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();               //키 입력을 받아 들임

     void AI();                    //컴퓨터의 행동


     Unit(int x, int y);

};


여기에 추가를 했으니 밑에 정의를 해야 겠죠.
일단 AI는 엄청 간단하게 만들 겠습니다.
현재 격투 판정은 만들지도 않았으니 단순히 쫓기만 하겠습니다.

앗! 그런데 여기서 문제가 생겼습니다!
어떻게 적 Unit 안에서 주인공의 x, y좌표를 알아내죠?
방법이 없습니다.

이건 설계미스입니다.
해결 방법은 (1)주인공의 x, y좌표를 전역변수로 선언한다.
(2)AI 부분은 메인 루프에서 처리한다.

(1)은 정말 마음에 안드는 군요.
그럼 (2)를 선택하기로 하겠습니다.
그런데 AI()만들어 놓은 걸 어쩌죠?
제가 설계 미스했습니다. 지우세요. 이러기엔 무책임하군요.
그럼 이렇게 바꿔 봅시다.

void AI(int hero_x, int hero_y);

네, 주인공의 위치를 변수로 받아서 처리하는 함수입니다.
다시 돌아가서 AI의 내용을 정의해 봅시다.

void Unit::AI(int hero_x, int hero_y)
{
     //가로 추적
     if(hero_x < x)x-=5;
     else if(hero_x > x)x+=5;

     //세로추적
     if(hero_y < y)y-=5;
     else if(hero_y > y)y+=5;
}

간단하게 자신의 위치와 주인공의 위치를 비교해보고
추적하는 방식입니다.

자, 그럼 이제 메인 루프 부분에서

hero.Control();

이 뒤에다

enemy.AI(hero.x, hero.y);

라고 추가해줍니다.
어떻게 됩니까? 에러나죠?
어째서 에러가 나냐고 보면 hero.x와 hero.y 때문입니다.
이 두 개는 전에 말했듯이 private: 로 선언된 녀석들이기 때문입니다.
'이건 나의 프라이버시다! 가르쳐 줄 수 없다!' 이거죠.
해결 방법은 일단 x,y를 public: 으로 바꿔 주는 겁니다.
근데 이건 맘에 안 듭니다.
일단 아래와 같은 것을 Unit 클래스 선언부에 추가합시다.

int GetX();
int GetY();

드디어 void가 아닌 리턴타입이 int인 메소드가 등장했군요.
지금까지 void는 아무 값도 안 돌려주고 그냥 일을 끝내는 메소드라면
앞에 리턴 함수가 붙는 메소드는 안에서 뭔가 처리를 한 후 특정 값을 돌려주는 함수입니다.
우린 뭐 특별히 연산 할 것 없이 이렇게 정의해 주면 됩니다.

int Unit::GetX()
{
     return x;
}

int Unit::GetY()
{
     return y;
}

별것 없습니다.

return x;

자기가 가진 x 값을 결과값으로 돌려 줍니다.

return y;

자기가 가진 y값을 결과값으로 돌려 줍니다.

이것을 이용해서 이전의 AI()부분을 바꿔 줍시다.

AI(hero.GetX(), hero.GetY());

지금쯤이면 이 사람 왜 이렇게 뻘짓을 하고 있지?
그냥 public으로 바꾸었으면 될 거 아냐?
라고 생각하시는 분이 계실지도 모르겠습니다.

이런 일을 하는 이유는 x,y를 내가 정한 방법 이외에는 접근하지 못하게 하기 위해서입니다.
이곳 저곳에서 x를 마구 불러내면 편하기는 하겠지만
일단 GetX를 통해서만 x값을 얻는 것이 체계가 잡혀있기 때문입니다.

현재 GetX만 존재하는 한 x값을 직접 바꾸는 방법은 없습니다.
초기화시에 자리 배치하는 것과, 주인공의 경우 컨트롤에 의한 이동, 적의 경우 AI에 의한 이동 외에는
갑자기 어디서 x를 바꾸는 일이 없습니다.
처음에는 쓸데없이 코드를 많이 쓰는 것 같아 불편하겠지만
나중에 소스의 규모가 커질수록 이렇게 만드는 것이 이해하기도 빠르고
버그가 생겼을 때 경로를 추적하기 좋습니다.

일단 지금까지 친 것을 실행시켜 보면 적이 너무 빨리 쫓아 옵니다.
주인공은 느려서 피할 수 없습니다.
아무래도 적의 움직임을 느리게 해야 할 듯 싶습니다.
근데 먼저 제가 전에 실수한 것을 바로 잡겠습니다.
Control() 부분에서

if(GetKey(vkey_up,1))

라고 했는데 GetKey의 두번째 파라미터 값을 0으로 바꿔 주십시오.

if(GetKey(vkey_up,0))

위와 같이 말이죠.
사실 이걸 1로 하면 1/100초를 쉰다음에 다음 입력을 받기 때문에 연속된 입력을 못 받습니다.
0으로 해야 하는 건데 제가 실수 했습니다.
아무튼 이렇게 해보면 적과 속도가 같지만 결국엔 따라잡힐 겁니다.
어쩔 수 없죠.

그럼 적의 AI 부분에서 속도를 바꿔 봅시다.

 //가로 추적
 if(hero_x < x)x-=3;
 else if(hero_x > x)x+=3;

 //세로추적
 if(hero_y < y)y-=3;
 else if(hero_y > y)y+=3;

이제 해보면 적을 피해다닐 수 있을 겁니다.
근데 맘에 안 듭니다.
왜 4개나 되는 수를 모두 바꿔야 되죠?
속도를 5에서 3으로 바꾸는데 매번 4개를 바꿔야 됩니다.
실수로 삑사리 나서 하나를 2로 적어다간 이상하게 움직일 겁니다.
언제나 4개가 동일한 값일테니 이건 변수로 해버립시다.
speed 변수를 선언하는 겁니다.

일단 Unit 클래스에 변수를 선언합니다.

int speed;

물론 접근하지 못하게 private: 부분에 말이죠.
그리고 추적부분을 바꿉시다.

 //가로 추적
 if(hero_x < x)x-=speed;
 else if(hero_x > x)x+=speed;

 //세로추적
 if(hero_y < y)y-=speed;
 else if(hero_y > y)y+=speed;

또 한 군데 바꿀 곳이 있죠? 주인공의 속도입니다.
Control도 바꿔 버립시다.

if(GetKey(vkey_up,0))
{
     y-=speed;
}
if(GetKey(vkey_down,0))
{
     y+=speed;
}
if(GetKey(vkey_left,0))
{
     x-=speed;
}
if(GetKey(vkey_right,0))
{
     x+=speed;
}

음, 좋습니다.
이제 마지막으로 speed를 설정할 수 있게 해야겠군요.
어디서 할 까요?
생성자 밖에 더 있나요?
생성자를 한번 더 바꿉시다.

Unit(int x, int y, int speed);

이렇게 말이죠.

this->speed = speed;

식으로 값을 받아주는 코드를 추가하고요.
그리고 생성할 때 주인공과 적의 속도를 다르게 적어 줍니다.

 Unit hero(300, 200, 5);
 Unit enemy(100, 300, 3);

이것으로 속도를 바꿔 보고 싶을 때는 저 숫자 하나만 바꾸면 됩니다.
만약 여러 종류의 타입의 적이 나온다면 저 값을 다르게 해서 설정해 주면 됩니다.

이것 저것 바꿔보면서 여러가지를 해보십시오.
내일은 동작을 고치든지 적 여러명을 나오게 하든지 하겠습니다.
혹시 먼저 배우고 싶은 게 있다면 댓글 적어 주십시오.

조회 수 :
3480
등록일 :
2008.04.27
19:15:42 (*.193.78.73)
엮인글 :
게시글 주소 :
https://hondoom.com/zbxe/index.php?mid=study&document_srl=193459
List of Articles
번호 제목 글쓴이 조회 수 추천 수 날짜 최근 수정일sort
165 Windows To Go와 R-Studio를 이용한 손실된 데이터 복구하기 노루발 89   2020-01-30 2020-01-30 19:52
하기의 복구 방법으로 모든 자료를 100% 복구할 수는 없으므로 자료의 손실 이전에 신뢰성 있고 주기적인 백업이 선행되어야 한다. 준비물: - 부팅 USB 혹은 외장 HDD/SSD - 복구한 자료를 저장할 외부 저장장치 (옵션) - R-Studio https://www.r-studio.com/ ...  
164 Love2d DPI 이슈 해결 [3] 노루발 97   2019-06-29 2019-07-01 06:34
이런 love 프로젝트가 있다고 하자. (conf.lua) function love.conf(t) t.window.width = 640 t.window.height = 360 end 창 크기를 640*480으로 설정한 뒤 실행하면 어떻게 보일까? 당연히 창 크기가 640*480 크기로 보여야겠지만 내 컴퓨터에서는 이렇게 보...  
163 Love2d 게임 중간에 광고 표시 [1] 노루발 377   2015-11-12 2015-11-17 00:16
http://love2d.org/forums/viewtopic.php?f=11&t=81224  
162 김프로 이미지 맵 만들기 노루발 393   2015-11-11 2015-11-11 08:05
https://docs.gimp.org/en/plug-in-imagemap.html  
161 RPG Maker MV 로컬라이징 방법 file 똥똥배 917   2015-10-27 2015-10-27 05:51
 
160 [번역] gamedev레딧의 Getting Started 문서 번역 [5] priling 1892   2014-12-26 2018-07-24 10:33
처음인 분들을 위한 '게임만들기' 가이드이 글은 [레딧 게임개발 커뮤니티의 /u/LordNed님의 포스팅]을 베이스로 작성한 것입니다. 이 글의 목적은 게임을 만들고 싶어하는 분들이 어떻게 시작할 수 있을지에 대해 명확한 가이드라인을 보여드리는 것입니다.  ...  
159 Love2d 안드로이드 게임 패키징하기 [3] 노루발 548   2014-12-15 2021-01-11 12:11
이 문서는 개발 환경이 갖추어져 있는 상태이고, 빌드를 무사히 마친 뒤라고 가정합니다. 또한 이 문서는 https://bitbucket.org/MartinFelis/love-android-sdl2/wiki/Game%20Packaging#markdown-header-how-to-package-the-apk-with-your-own-love-090-game ...  
158 Love2d 안드로이드 빌드하기 노루발 572   2014-12-15 2014-12-15 00:59
Love2d의 안드로이드 포트는 알파 단계입니다. 차차 개선되어 나가긴 하겠지만 아직 불안정한 부분이 많으며, 일어날 수 있는 오작동과 그로 인한 결과는 일절 책임지지 않습니다.   이 문서는 Windows 사용자 기준입니다. Linux나 Mac을 사용할 정도의 내공...  
157 Love2d 여러 플랫폼으로 빌드 자동화 노루발 470   2014-11-12 2014-11-12 17:35
https://github.com/MisterDA/love-release/blob/master/README.md http://www.ambience.sk/lua-love2d-game-distribution/  
156 구글 인앱 구매 Soomla로 구현해본 후 팁 똥똥배 577   2014-09-20 2014-09-20 18:45
1. 일단 알파 테스트라도 앱을 출시시켜야 인앱이 작동한다. 그리고 APK 업로드 후에 바로 인앱이 적용되지 않는다. 구글에 게시되는 데, 시간이 걸리므로 인앱 등록 - APK 등록을 마친 후 다음날부터 구현하는 게 깔끔하다. 이걸로 모르고 하루종일 왜 안되나...  
155 cocos2d-x 2.2.2 문자열 출력 버그 [2] 똥똥배 601   2014-06-10 2014-08-23 22:35
cocos2d-x 2.2.2, 맥에서만 나타는 문제다. 윈도우, 안드로이드에서는 아무 이상 없다. 줄 수가 꽤 긴 문자열을 출력할 경우, 문자가 전부 다 나오지 않고 잘린다. 잘리는 기준은 줄이다. 문자 수가 어떻든 간에 줄 수에 따라서 잘려버린다. 아마 문자열의 줄 ...  
154 cocos2d-x 2.2.2 윈도우 환경 기본 메모리 누수 똥똥배 640   2014-03-10 2014-03-11 03:51
cocos2d-x 2.2.2를 윈도우에서 실행시키고 나서 종료하면 기본으로 4바이트의 메모리 누수가 발생한다. 이건 CCScriptEngineManager가 원인인데 CCObject들은 delete를 실행할 때마다 CCScriptEngineManager의 sharedManager를 호출하기 때문이다. 결국 모든 C...  
153 cocos2d-x 2.2.2 UILabelBMFont 메모리 누수 해결법 [2] 똥똥배 783   2014-03-10 2014-05-31 19:30
만약 CocoStudio로 UI를 만든 후 GUIReader를 써서 지정 폰트를 불러오면 엔진 버그로 인해서 메모리 누수가 생긴다. 이걸 고치려면 LabelBMFont.cpp를 아래와 같이 수정해야 한다. void LabelBMFont::setFntFile(const char *fileName) { if (!fileName || st...  
152 Cocostudio의 ActionNode 메모리 누수 해결법(cocos2d-x 2.2.2) [2] 똥똥배 776   2014-03-09 2014-05-31 19:26
Cocostudio로 액션을 만들고 액션매니저를 이용해서 ActionManager::shareManager()->playActionByName 아래와 같이 실행을 해주면 액션이 실행되는 경우 메모리 누수가 발생한다. (액션을 실행하지 않았을 때는 메모리 누수 없음) 이 메모리 누수를 없애려면 ...  
151 CCTextFieldTTF 0바이트 메모리 누수 버그 해결법 똥똥배 692   2014-01-13 2014-01-13 10:38
CCTextFieldTTF::textFieldWithPlaceHolder("", FONT_NAME, FONT_SIZE); 이런 식으로 텍스트필드를 생성하면 ""가 0바이트라서 해체를 못하고 0바이트 누수가 발생한다. 그냥 스페이스(" ")라도 넣어주면 누수는 발생하지 않는다. 이 버그가 발생하는 버전은 2...  
150 Lua 소코반 EX: 포팅: 3 (머나먼 여행길 안녕 친구여) 노루발 913   2013-09-18 2013-09-18 08:05
안녕하세요, 노루발입니다. 오늘로 그래픽 소코반을 끝내기로 했습니다. 아쉽지만 끝내야죠. 일단 걸음수와 이것저것을 뿌려주기로 합니다. 그런데 이미 화면이 꽉 차버려서 뿌려줄 공간이 없네요... 없으면 만들면 되지. 창을 늘립시다. conf.lua를 수정. fun...  
149 Lua 소코반 EX: 포팅: 2 (키가 눌리면 이동하게 해 보자) 노루발 832   2013-09-18 2013-09-18 08:04
-- 플레이어 위치를 저장하는 변수 -- 초기 시작지점은 5, 5로 설정. playerx = 5 playery = 5 -- 박스의 위치를 저장하는 배열. box = {} -- 박스 목표 지점의 위치를 저장하는 배열. goal = {} -- 움직인 횟수 저장 moves = 0 -- 맵 데이터를 저장하는 8*8의...  
148 Lua 소코반 EX: 포팅: 1 노루발 856   2013-09-18 2013-09-18 08:03
-- 플레이어 위치를 저장하는 변수 -- 초기 시작지점은 5, 5로 설정. playerx = 5 playery = 5 -- 박스의 위치를 저장하는 배열. box = {} -- 박스 목표 지점의 위치를 저장하는 배열. goal = {} -- 움직인 횟수 저장 moves = 0 -- 맵 데이터를 저장하는 8*8의...  
147 Lua 소코반 EX: 그래픽 준비 노루발 1084   2013-09-18 2013-09-18 08:00
안녕하세요, 노루발입니다. 얼마 전 만든 Lua 소코반을 게임다운 게임을 만들기 위해 그래픽을 씌워보려고 합니다. 저번에 한 Love2D로 살펴본 뭐가 뭔지 모르는 복잡한 그리기 함수들도 사실은 모두가 소코반을 그래픽으로 만들기 위한 훼이크였습니다. 후후....  
146 게임의 기본 설정을 담당하는 love.conf 노루발 763   2013-09-17 2013-09-17 08:39
안녕하세요, 노루발입니다. 여태까지 이것저것 많이 Love2D를 다뤄와서 제가 만든 소코반에 그래픽을 씌울 수준까지 되었지만 (물론 키 입력은 논외로 치죠, 나중에 다룰게요.) 아직 이걸로 정식 게임을 만든다면 모자라도 한참 모자랍니다. 창의 이름도 실행...