실습4:
교재 6.동시처리 챕터에서 페이지 68-67사이 select 이용한 채팅 서버 -클라이언트 구현
chatting_server.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
|
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/time.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define BUFSIZE 100
int main(int argc, char **argv)
{
int serv_sock;
struct sockaddr_in serv_addr;
fd_set reads, temps;
int fd_max;
char message[BUFSIZE];
int str_len;
struct timeval timeout;
if (argc != 2)
{
printf("Usage : %s <port>\n", argv[0]);
return -1;
}
int so_reuseaddr = 1;
int state = setsockopt(serv_sock, SOL_SOCKET, SO_REUSEADDR,
(void *)&so_reuseaddr, sizeof(so_reuseaddr));
serv_sock = socket(PF_INET, SOCK_STREAM, 0);
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
serv_addr.sin_port = htons(atoi(argv[1]));
if (bind(serv_sock, (struct sockaddr *)&serv_addr,
sizeof(serv_addr)) == -1)
{
perror("bind() error");
return -1;
}
if (listen(serv_sock, 5) == -1)
{
perror("linsten() error");
return -1;
}
FD_ZERO(&reads);
FD_SET(serv_sock, &reads); // 파일 디스크립터 설정(서버)
fd_max = serv_sock;
while (1)
{ // 무한 루프
int clnt_fd, fd, str_len;
int clnt_sock, clnt_len;
struct sockaddr_in clnt_addr;
temps = reads; // select 함수 변화가 생긴 디스크립터를
// 제외한 나머지 비트들을 0으로 초기화
timeout.tv_sec = 5; // 초 단위 설정
timeout.tv_usec = 0; // 마이크로 초 단위 설정
if (select(fd_max + 1, &temps, 0, 0, &timeout) == -1)
{
perror("select() error");
return -1;
}
for (fd = 0; fd < fd_max + 1; fd++)
{
if (FD_ISSET(fd, &temps))
{
if (fd == serv_sock)
{
clnt_len = sizeof(clnt_addr);
clnt_sock = accept(serv_sock,
(struct sockaddr *)&clnt_addr, &clnt_len);
FD_SET(clnt_sock, &reads);
if (fd_max < clnt_sock)
fd_max = clnt_sock;
printf("클라이언트 연결 : FD(%d) \n", clnt_sock);
}
else
{
str_len = read(fd, message, BUFSIZE);
if (str_len == 0)
{
FD_CLR(fd, &reads);
close(fd);
printf("클라이언트 종료 : FD(%d)\n", fd);
}
else
{
for (clnt_fd = serv_sock + 1; clnt_fd < fd_max +
1; clnt_fd++)
write(clnt_fd, message, str_len);
}
}
}
}
}
return 0;
}
http://colorscripter.com/info#e" target="_blank" style="color:#4f4f4f; text-decoration:none">Colored by Color Scripter
|
http://colorscripter.com/info#e" target="_blank" style="text-decoration:none; color:white">cs |
chatting_client.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
#include <stdio.h>
#include <fcntl.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/time.h>
#include <string.h>
#define MAXLINE 512
#define MAX_SOCK 128
char name[10];
int main(int argc, char *argv[])
{
int s;
struct sockaddr_in server_addr;
int n, pid;
int maxfdp1;
fd_set read_fds;
char line[MAXLINE];
char message[MAXLINE + 1];
if (argc != 4)
{
printf("사용법 : %s sever_IP port name \n", argv[0]);
return -1;
}
sprintf(name, "[%s]", argv[3]);
if ((s = socket(PF_INET, SOCK_STREAM, 0)) < 0)
{
printf("Client : Can't open stream socket.\n");
return -1;
}
bzero((char *)&server_addr, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = inet_addr(argv[1]);
server_addr.sin_port = htons(atoi(argv[2]));
if (connect(s, (struct sockaddr *)&server_addr, sizeof(server_addr))
< 0)
{
printf("Client : Can't connect to server.\n");
return -1;
}
else
{
printf("서버에 접속되었습니다. \n");
}
maxfdp1 = s + 1;
FD_ZERO(&read_fds);
while (1)
{
FD_SET(0, &read_fds);
FD_SET(s, &read_fds);
if (select(maxfdp1, &read_fds, (fd_set *)0,
(fd_set *)0, (struct timeval *)0) < 0)
{
printf("select error\n");
return -1;
}
if (FD_ISSET(s, &read_fds))
{
int size;
if ((size = recv(s, message, MAXLINE, 0)) > 0)
{
message[size] = '\0';
printf("%s \n", message);
}
}
if (FD_ISSET(0, &read_fds))
{
fgets(message, MAXLINE, stdin);
if (!strcmp(message, "q\n"))
{
close(s);
return -1;
}
sprintf(line, "%s %s", name, message);
send(s, line, strlen(line), 0);
}
}
return 0;
}
http://colorscripter.com/info#e" target="_blank" style="color:#4f4f4f; text-decoration:none">Colored by Color Scripter
|
http://colorscripter.com/info#e" target="_blank" style="text-decoration:none; color:white">cs |
'딥러닝 기반 영상인식 개발 전문가 과정 > 리눅스' 카테고리의 다른 글
6월14일 epoll, 레벨트리거, 엣지트리거, 멀티캐스트 (0) | 2019.06.14 |
---|---|
6월14일 채팅실습, fileno, fdopen, 입력 스트림과 출력 스트림의 분리 (0) | 2019.06.14 |
6월13일 실습3, 표준 입출력 함수, send,recv (0) | 2019.06.13 |
6월13일 실습, IO Multiplexing, select (0) | 2019.06.13 |
6월12일 프로세스 기반 다중접속 서버 모델, TCP, 소켓의 다른 옵션 (0) | 2019.06.12 |