게으른 개발자의 끄적거림

소켓(SOCKET)통신 이란?

끄적잉 2024. 6. 19. 21:39

## 소켓 통신 개요

 

 


### 소켓이란?

소켓(Socket)은 네트워크 상에서 통신을 하기 위한 종단점(endpoint)입니다. 소켓은 프로세스 간 통신을 위해 필요하며, 클라이언트와 서버가 데이터를 주고받는 데 사용됩니다. 소켓은 운영체제에 의해 관리되며, IP 주소와 포트 번호를 통해 식별됩니다. 네트워크 프로그래밍에서 소켓을 사용하면 TCP/IP, UDP 등 다양한 프로토콜을 기반으로 통신할 수 있습니다.

 


### 소켓의 종류

1. **스트림 소켓(Stream Socket)**: TCP(Transmission Control Protocol)를 사용하여 신뢰성 있는 데이터 전송을 제공합니다. 데이터는 순서대로 전송되고, 손실이 발생하면 재전송됩니다. 대부분의 네트워크 애플리케이션은 스트림 소켓을 사용합니다.

2. **데이터그램 소켓(Datagram Socket)**: UDP(User Datagram Protocol)를 사용하여 비신뢰성 데이터 전송을 제공합니다. 데이터는 개별적인 데이터그램으로 전송되며, 순서 보장이나 재전송 기능이 없습니다. 실시간 전송이 중요한 경우(예: 스트리밍, 온라인 게임) 사용됩니다.

 

 


### 소켓 통신의 과정

소켓 통신은 일반적으로 다음과 같은 단계로 이루어집니다.

1. **소켓 생성**: 통신에 필요한 소켓을 생성합니다.
2. **서버 바인드**: 서버는 특정 포트에 소켓을 바인딩하여 클라이언트 요청을 대기합니다.
3. **서버 리슨**: 서버는 클라이언트의 연결 요청을 대기합니다.
4. **클라이언트 연결**: 클라이언트는 서버에 연결 요청을 합니다.
5. **연결 수락**: 서버는 클라이언트의 연결 요청을 수락합니다.
6. **데이터 전송**: 클라이언트와 서버 간에 데이터를 주고받습니다.
7. **연결 종료**: 통신이 끝나면 소켓을 닫습니다.

 


### 소켓 프로그래밍

소켓 프로그래밍은 다양한 프로그래밍 언어에서 지원됩니다. 여기서는 Python을 예로 들어 소켓 프로그래밍을 설명하겠습니다.

 

 


#### Python 소켓 프로그래밍

Python에서 소켓 프로그래밍을 하기 위해서는 `socket` 모듈을 사용합니다. 다음은 간단한 클라이언트-서버 예제입니다.

 


**서버 코드:**

```python
import socket

# 소켓 생성
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)


# 소켓 바인드
server_socket.bind(('localhost', 12345))

# 연결 대기
server_socket.listen(5)
print("서버가 시작되었습니다. 연결을 기다리는 중입니다...")

# 연결 수락
client_socket, addr = server_socket.accept()
print(f"{addr}에서 연결되었습니다.")

# 데이터 수신
data = client_socket.recv(1024)
print(f"클라이언트로부터 받은 데이터: {data.decode()}")

# 데이터 송신
client_socket.sendall("Hello, Client!".encode())

# 소켓 닫기
client_socket.close()
server_socket.close()
```

 

 


**클라이언트 코드:**

```python
import socket

# 소켓 생성
client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 서버에 연결
client_socket.connect(('localhost', 12345))

# 데이터 송신
client_socket.sendall("Hello, Server!".encode())

# 데이터 수신
data = client_socket.recv(1024)
print(f"서버로부터 받은 데이터: {data.decode()}")

# 소켓 닫기
client_socket.close()
```


### 소켓 프로그래밍의 주요 함수

1. **socket()**: 소켓을 생성합니다. 인자로 주소 체계(AF_INET)와 소켓 타입(SOCK_STREAM 또는 SOCK_DGRAM)을 지정합니다.
2. **bind()**: 소켓에 IP 주소와 포트 번호를 할당합니다.
3. **listen()**: 서버 소켓이 클라이언트의 연결 요청을 기다리게 합니다.
4. **accept()**: 클라이언트의 연결 요청을 수락합니다. 새로운 소켓 객체와 클라이언트의 주소를 반환합니다.
5. **connect()**: 클라이언트 소켓이 서버에 연결을 요청합니다.
6. **send()**: 데이터를 전송합니다.
7. **recv()**: 데이터를 수신합니다.
8. **close()**: 소켓을 닫습니다.

 


### 소켓 통신의 주요 이슈

1. **동기화 문제**: 클라이언트와 서버 간의 데이터 전송 순서를 맞추는 것이 중요합니다.
2. **네트워크 지연**: 네트워크 상태에 따라 지연이 발생할 수 있으며, 이는 데이터 전송에 영향을 미칠 수 있습니다.
3. **보안 문제**: 데이터를 암호화하여 전송하거나, SSL/TLS를 사용하여 보안을 강화할 수 있습니다.
4. **포트 충돌**: 동일한 포트를 사용하는 여러 애플리케이션이 있을 경우 충돌이 발생할 수 있습니다.

 


### 소켓 통신의 응용 분야

소켓 통신은 다양한 응용 분야에서 사용됩니다.

1. **웹 서버**: HTTP 프로토콜을 사용하여 웹 브라우저와 통신합니다.
2. **파일 전송**: 파일을 클라이언트와 서버 간에 전송합니다.
3. **채팅 애플리케이션**: 실시간 메시징을 제공합니다.
4. **온라인 게임**: 게임 클라이언트와 서버 간의 실시간 통신을 처리합니다.
5. **IoT 기기 통신**: 센서와 기기 간의 데이터 교환을 처리합니다.

 


### 고급 소켓 프로그래밍

고급 소켓 프로그래밍에서는 비동기식 통신, 멀티스레딩, 멀티프로세싱 등을 사용할 수 있습니다.

#### 비동기식 소켓 프로그래밍

비동기식 소켓 프로그래밍은 `select` 모듈을 사용하여 구현할 수 있습니다. 이 방식은 여러 소켓을 동시에 처리할 수 있게 해줍니다.

```python
import socket
import select

# 소켓 생성 및 바인드
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_socket.bind(('localhost', 12345))
server_socket.listen(5)

sockets_list = [server_socket]
clients = {}

print("서버가 시작되었습니다. 연결을 기다리는 중입니다...")

while True:
    read_sockets, _, exception_sockets = select.select(sockets_list, [], sockets_list)

    for notified_socket in read_sockets:
        if notified_socket == server_socket:
            client_socket, client_address = server_socket.accept()
            sockets_list.append(client_socket)
            clients[client_socket] = client_address
            print(f"{client_address}에서 연결되었습니다.")
        else:
            message = notified_socket.recv(1024)
            if message:
                print(f"클라이언트로부터 받은 데이터: {message.decode()}")
                notified_socket.sendall("메시지 수신 완료".encode())
            else:
                sockets_list.remove(notified_socket)
                del clients[notified_socket]

    for notified_socket in exception_sockets:
        sockets_list.remove(notified_socket)
        del clients[notified_socket]
```


### 결론

소켓 통신은 네트워크 프로그래밍의 핵심 요소로, 다양한 응용 프로그램에서 필수적으로 사용됩니다. 소켓 프로그래밍을 통해 클라이언트-서버 모델을 구현하고 데이터를 주고받을 수 있습니다. 기본적인 소켓 프로그래밍부터 고급 비동기식 소켓 프로그래밍까지 다양한 기술을 익히면, 네트워크 기반 애플리케이션을 효율적으로 개발할 수 있습니다. 소켓 통신의 이해와 활용은 현대의 분산 시스템 및 네트워크 애플리케이션 개발에 있어 중요한 역할을 합니다.