본문 바로가기

[ Windows Program ]/Windows API

비동기소켓 서버구현



그림이 작으니 클릭해서 큰 그림으로 보세요

 

 

우리가 만들어 볼 프로그램의 스샷입니다. '서버'와 '클라이언트' 프로그램으로 나뉘어서 따로 구현하게 될겁니다. 이번 장에서는 서버 부분만 알아볼텐데요. 소스는 VC++ 6.0 프로젝트 파일을 압축해서 올렸습니다.

 

이 글에서는 전체 코드는 다루지 않고 중요한 몇부분만 설명드릴테니 완성코드는 직접 다운받아서 보시고 이해는 여기 글 보면서 하세요. 시작합니다~

 

 

먼저 이 채팅프로그램은 1:1 형식의 프로그램이 아니라 1:다수 형식으로 만들어 졌습니다. 그리고 클라이언트의 데이터는 모두 서버로 전송되게 만들어져 있습니다. 쉽게 말해서 서버가 하나 열려있고 3명의 클라이언트가 그 서버에 접속한 상태라고 가정을 하면 1번 클라이언트가 채팅메시지를 전송할때 직접 2번, 3번 클라이언트에게 하나씩 보내주는것이 아니라 서버에게만 채팅메시지를 전달하고 서버가 다시 1번, 2번, 3번 클라이언트에게 채팅메시지를 전달하는 방식입니다.

 

그래서 당연히 서버측에서는 클라이언트 소켓을 배열로 여러개 만들어놓고 관리하는데 최대 32명의 클라이언트까지 접속할 수 있도록 했습니다.

 

#define MAXCLIENT           32

 

SOCKET g_servSock;

SOCKET g_clntSock[MAXCLIENT];

int g_index = 0;

 

서버소켓은 하나를 생성했으며 서버소켓이 받아 들일 수 있는 최고 클라이언트 수가 32입니다. 클라이언트 소켓이 여러개라 배열로 선언했으며 g_index에는 현재 클라이언트의 접속자 수가 들어갑니다. 접속할때마다 수가 증가하고 연결해제할때마다 감소하겠죠?

 

WinMain 함수내에서는 메인 다이얼로그창을 생성합니다.

아래는 다이얼로그 메시지 프로시저의 일부 입니다.

 

case WM_INITDIALOG:

    g_servSock = InitSocket( hDlg );

    if( INVALID_SOCKET == g_servSock )

        EndDialog( hDlg, 0 );

 

여기에서 InitSocket 이란 사용자 정의함수를 호출하는데 이 함수가 하는일은 서버소켓을 생성하고, bind시키고 listen 시킨뒤에 WSAAsyncSelect함수로 비동기소켓으로 전환하고 그 해당 소켓을 리턴하는 함수입니다.

 

아래는 비동기 메시지 처리부분입니다.

 

#define WM_ASYNC          (WM_USER + 1)   // 비동기 메시지 정의

 

...

 

case WM_ASYNC:

    switch( WSAGETSELECTMESSAGE(lParam) )

    {

    case FD_ACCEPT:

        // 클라이언트의 접속요청을 처리

        break;

 

    case FD_READ:

        // 클라이언트가 데이터를 보내왔을때의 처리

        break;

 

    case FD_CLOSE:

        // 클라이언트가 연결을 해제했을때의 처리

        break;

    }

 

 

FD_ACCEPT 부분을 먼저 살펴보면..

 

// 먼저 현재접속된 클라이언트의 수를 체크하여 최고치라면 그냥 무시한다

if( MAXCLIENT - 1 <= g_index )

    break;

// 연결을 수락한다

g_clntSock[g_index] = OnAccept( hDlg, g_servSock );

 

...

 

OnAccept 핸들러 함수를 호출해주고 있는데 이 함수에서는 클라이언트 소켓을 생성하고 비동기소켓으로 전환하는 역할을 합니다.

 

- OnAccept 부분

SOCKET sock;

SOCKADDR_IN Addr;

int iLen;

 

// 접속을 수락하기 위해 accept를 호출

sock = accept( ... );

 

// 클라이언트 소켓을 비동기소켓으로 전환

WSAAsyncSelect( sock, hWnd, WM_ASYNC, FD_READ | FD_CLOSE );

 

return sock;

 

 

FD_READ 부분

FD_READ메시지가 도착했을땐 먼저 데이터를 받은뒤에 그 받은 데이터를 접속해 있는 모든 클라이언트에게 뿌려줍니다. 여기서 중요한점은 FD_READ 메시지가 발생되면 반드시 recv함수를 호출해서 데이터를 받아야 하는데 recv함수의 첫번째 인수엔 서버소켓을 대입하면 안됩니다. 이때는 FD_READ메시지를 일으킨 클라이언트 소켓을 대입해줘야 합니다.

 

비동기 메시지가 일어나면 wParam에 비동기 메시지를 일으킨 소켓의 식별자가 대입되어 들어오는데 반드시 이 값을 넘겨주어야 합니다. 이 점 꼭 주의하세요 !!

 

case FD_READ:

    OnRead(SOCKET)wParam, g_clntSock, g_index );

    break;

 

...

 

OnRead 함수부분

...

if( SOCKET_ERROR != recv( sock, Msg, 128, 0 ) )

{

    for( i = 0; i < index; i++ )

        send( sockArray[i], Msg, strlen( Msg ) + 1, 0 );

}

...

 

 

FD_CLOSE 부분

이 부분에선 연결해제 메시지를 보내온 소켓을 닫아주고 배열을 재 조정해주는 OnClose 핸들러 함수를 호출합니다.

 

case FD_CLOSE:

    OnClose( g_clntSock, (SOCKET)lParam );

    g_index--;

    ....

 

OnClose 함수부분

void OnClose( SOCKET *sockArray, SOCKET sock )

 

...

 

for( i = 0; i < MAXCLIENT; i++ )

{

    if( sock == sockArray[i] )

    {

        // 연결해제 메시지를 보내온 소켓을 찾아 소켓을 닫아준다

        closesocket( sockArray[i] );

 

        // 소켓배열 가운데가 비기때문에 한칸씩 당겨주어서 배열을 재조정한다.

        for( j = i; j < MAXCLIENT - 1; j++ )

            sockArray[j] = sockArray[j + 1];

    }

}

 

 

여기 설명한 부분에서는 에러처리 같은 여러가지를 생략하고 작성했어요.

다시 한번 말씀드리지만 꼭 소스를 다운받아서 분석해 보셔야합니다. 그리고 물론 직접 만들어 봐야겠죠? ^^