RadarURL

응용 프로그래밍
2009.05.30 22:20

유한상태기계(FSM: finite state machine)

조회 수 8568 추천 수 0 댓글 0
?

단축키

Prev이전 문서

Next다음 문서

크게 작게 위로 아래로 댓글로 가기 인쇄
?

단축키

Prev이전 문서

Next다음 문서

크게 작게 위로 아래로 댓글로 가기 인쇄
유한상태기계(finite state machine, FSM)는 인정받은 설계 기법입니다. 유한상태기계는 상태(state)들의 유한한 집합이며, 어떤 때에 오직 하나의 상태만 활성화되고, 그 상태에 대한 코드가 실행됩니다. 어떤 상태에서 다른 상태로 전환하는 방법은 두 가지가 있는데요. 첫 번째는 어떤 상태에 대한 코드가 다른 상태로 바꿔야 하는 상황을 검사하여 그 상태로 전환하는 것입니다. 이 때는 상태에 따른 행동, 상태 전환 코드가 그 객체에만 들어있게 됩니다. 두 번째 방법에서는 상태 관련 코드의 밖에 있는 코드에서 상황들을 점검하고 상태를 전환합니다. 이 때는 상태에 따른 행동은 그 객체에 들어있고 상태 전환 코드는 다른 객체(주로 관리자 객체)에 있거나, 아니면 둘 다 관리자 객체에 있습니다.

일인칭 슈팅 게임에서 주인공을 죽이는 것이 목적인 인공지능 적이 있다고 가정합시다. 적에게는 Patrol, Attack이라는 두 상태가 있습니다. Patrol 상태에서는 주인공을 탐색하는 코드만 실행하는데, 주인공을 발견하면 Attack 상태로 전환합니다. 이를 코드로 표현하면 다음과 같습니다. (그나마 제대로 할 줄 아는 언어가 액션스크립트밖에 없어서 액션스크립트로 씁니다)

///////////////////////////////////////////////
// 초기화 코드에서
const PATROL:String = "patrol";
const ATTACK:String = "attack";

var state:String = PATROL;

///////////////////////////////////////////////
// enterFrame 수신자 또는 타이머 함수 안에서
switch(state){
 case PATROL :
  // 정찰을 수행한다
  if(주인공을 발견했다면){
   state = ATTACK;
  }
  break;
 case ATTACK :
  // 주인공을 공격한다
  break;
}
///////////////////////////////////////////////


또는 행동 코드와 전환 코드를 분리하여 이렇게 쓸 수 있습니다.

///////////////////////////////////////////////
// 초기화 코드에서
const PATROL:String = "patrol";
const ATTACK:String = "attack";

var state:String = PATROL;

///////////////////////////////////////////////
// enterFrame 수신자 또는 타이머 함수 안에서
switch(state){
 case PATROL :
  // 정찰을 수행한다
  break;
 case ATTACK :
  // 주인공을 공격한다
  break;
}

///////////////////////////////////////////////
// 다른 함수 안에서
if(state == PATROL && 적이 주인공을 발견했다면){
 state = ATTACK;
}
///////////////////////////////////////////////

사실 각 상태에 따른 행동은 매우 복잡해질 수 있고 그에 따라 코드의 양도 늘어날 수 있으므로 상태 수행 코드를 어떤 함수에 따로 작성하고 switch문 안에서는 그 함수를 호출만 하는 식으로 코드를 작성할 수도 있습니다. (윈도우 API에서 메시지를 이렇게 처리하는 것을 메시지 크랙킹이라고 하던가요?)

///////////////////////////////////////////////
// 초기화 코드에서
const PATROL:String = "patrol";
const ATTACK:String = "attack";

var state:String = PATROL;

///////////////////////////////////////////////
// enterFrame 수신자 또는 타이머 함수 안에서
switch(state){
 case PATROL :
  patrol();
  break;
 case ATTACK :
  attack();
  break;
}

///////////////////////////////////////////////
// 함수 정의
function patrol():void {
 // 정찰을 수행한다 (코드 매우 많음)
 if(주인공을 발견했다면){
  state = ATTACK;
 }
}
function attack():void {
 // 주인공을 공격한다 (코드 매우 많음)
}

///////////////////////////////////////////////


기왕이면 함수 참조를 써서 state 변수랑 switch문을 없애버립시다.

///////////////////////////////////////////////
// 초기화 코드에서
const PATROL:String = "patrol";
const ATTACK:String = "attack";

var action:Function = patrol;

///////////////////////////////////////////////
// enterFrame 수신자 또는 타이머 함수 안에서
action();

///////////////////////////////////////////////
// 함수 정의
function patrol():void {
 // 정찰을 수행한다 (코드 매우 많음)
 if(주인공을 발견했다면){
  action = attack;
 }
}
function attack():void {
 // 주인공을 공격한다 (코드 매우 많음)
}

///////////////////////////////////////////////


유한상태기계는 여러 방법으로 응용할 수 있습니다. 두 가지만 예를 들어보자면 유한상태기계에 디자인 패턴을 적용하는 것과 스택 기반 유한상태기계를 사용하는 것입니다. 관련 자료를 참고하시기 바랍니다.

유한상태기계의 단점은 애초에 의도한 것 이상을 구현하기 어렵다는 점입니다. 확장이 어렵지요. 그래서 인공지능 쪽에서는 유한상태기계를 넘어서는 새로운 방법이 많이 연구되었습니다.



출처 : http://liverwort.tistory.com/458
?

공부 게시판

공부에 도움되는 글을 올려주세요.

List of Articles
번호 분류 제목 글쓴이 날짜 조회 수
공지 [공지] 공부 게시판 입니다. 처누 2003.08.18 931254
2503 유닉스/리눅스 리눅스 서버 설치 중 에러 "Failed to find a suitable stage1 device" JaeSoo 2025.10.03 37
2502 유닉스/리눅스 Linux 11 . Linux 설치 시 lvm 수동설정 JaeSoo 2025.10.03 28
2501 유닉스/리눅스 [Rocky Linux] 누구나 쉽게 따라하는 Rocky Linux 9.0 OS 다운로드 및 설치 방법~!! JaeSoo 2025.10.03 42
2500 유닉스/리눅스 Linux/Rocky Linux Rocky Linux : Composer 설치 JaeSoo 2025.09.29 43
2499 유닉스/리눅스 [Rocky Linux] 록키 리눅스 최신 업데이트 적용 방법 JaeSoo 2025.09.29 44
2498 유닉스/리눅스 rocky linux 커널 업데이트 JaeSoo 2025.09.29 36
2497 웹서버,WAS [Windows] OWASP ZAP 사용법 JaeSoo 2025.09.29 50
2496 웹서버,WAS [웹 취약점] 웹서버 디렉토리 리스팅 방지 JaeSoo 2025.09.29 52
2495 유닉스/리눅스 Clonezilla(클론질라) 백업파일 하드디스크로 복구 방법 JaeSoo 2025.09.28 44
2494 유닉스/리눅스 클론질라 사용 방법 - OS 이미지 백업 CloneZilla JaeSoo 2025.09.28 57
2493 유닉스/리눅스 [Linux / Rocky] FTP 접속 관련 설정 실습 예제 JaeSoo 2025.09.23 73
2492 유닉스/리눅스 Could not reliably determine the server's fully qualified domain name JaeSoo 2025.09.23 72
2491 유닉스/리눅스 [아파치] 아파치 웹서버에 .htaccess 파일 설정하는 방법 JaeSoo 2025.09.23 65
2490 유닉스/리눅스 [Linux] Apache web server의 rewrite module 사용하기 [출처] [Linux] Apache web server의 rewrite module 사용하기|작성자 g00dmoney JaeSoo 2025.09.23 61
2489 유닉스/리눅스 록키 리눅스 9.1에서 apache, mysql, PHP 설치 JaeSoo 2025.09.23 68
2488 유닉스/리눅스 [권한문제] 라이믹스 Fatal error: Uncaught Rhymix\Framework\Exception: Cannot write template cache file JaeSoo 2025.09.23 57
2487 유닉스/리눅스 라이믹스 php 에러 표시 JaeSoo 2025.09.23 53
2486 유닉스/리눅스 Rocky Linux - 라이믹스(Rhymix) 설치하기 JaeSoo 2025.09.23 50
2485 유닉스/리눅스 라이믹스(Rhymix) 매뉴얼 JaeSoo 2025.09.23 46
2484 유닉스/리눅스 [ Apache ] 아파치 start/stop/restart 명령어 JaeSoo 2025.09.23 37
Board Pagination Prev 1 2 3 4 5 6 7 8 9 10 ... 126 Next
/ 126


즐겨찾기 (가족)

JAESOO's HOMEPAGE


YOUNGAE's HOMEPAGE


장여은 홈페이지


장여희 홈페이지


장여원 홈페이지


즐겨찾기 (업무)

알리카페 홀릭

숭실대 컴퓨터 통신연구실 (서창진)

말레이시아 KL Sentral 한국인 GuestHouse


즐겨찾기 (취미)

어드민아이디

유에코 사랑회

아스가르드 좋은사람/나쁜사람

JServer.kr

제이서버 메타블로그

재수 티스토리


즐겨찾기 (강의, 커뮤니티)

재수 강의 홈페이지


한소리


VTMODE.COM


숭실대 인공지능학과


숭실대 통신연구실


베너