forked from dd-Dog/c-case-chatroom
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathclient.c
117 lines (103 loc) · 3.8 KB
/
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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#include "utility.h"
int main(){
int socketfd = 0;
unsigned int remoteAddr = 0;
struct sockaddr_in socketRemoteAddr = {0};
socklen_t socklen = 0;
char buf[4096] = {0};
socketfd = socket(AF_INET, SOCK_STREAM, 0);
if(0 > socketfd){
printf("创建Socket失败!\n");
return -1;
}
socketRemoteAddr.sin_family = AF_INET;
socketRemoteAddr.sin_port = htons(SERVER_PORT);
socketRemoteAddr.sin_addr.s_addr = inet_addr(SERVER_IP);
if(0 > connect(socketfd, (void *)&socketRemoteAddr, sizeof(socketRemoteAddr))){
printf("连接失败\n");
}else {
printf("连接成功\n");
//创建管道,其中fd[0]用于父进程读,fd[1]用于子进程写
int pipe_fd[2];
if(pipe(pipe_fd) < 0){
perror("pipe error");
exit(-1);
}
//创建epoll
int epfd = epoll_create(EPOLL_SIZE);
if(epfd < 0){
perror("epoll create error");
exit(-1);
}
static struct epoll_event events[2];
//将socket和管道描述符都添加到内核事件表中
addfd(epfd, socketfd, 1);
addfd(epfd, pipe_fd[0], 1);
int isClientWork = 1;//表示 客户端是否正常工作
char message[BUF_SIZE];
//创建子进程
int pid = fork();
if(pid < 0){
perror("forl error");
exit(-1);
}else if(pid == 0){
//子进程,pid=0表示子进程
//子进程负责写入管道,因此先关闭读端
close(pipe_fd[0]);
printf("Please input 'exit' to exit the chat room\n");
while(isClientWork){
bzero(&message, BUF_SIZE);
fgets(message, BUF_SIZE, stdin);//从控制台读取输入内容
//判断是否是'exit'
if(strncmp(message, EXIT, strlen(EXIT)) == 0){
isClientWork = 0;
} else {
//将消息写入管道
if(write(pipe_fd[1], message, strlen(message) - 1) < 0) {
perror("pipe write error");
exit(-1);
}
}
}
}else{
//父进程负责读管道数据,因此先关闭写端
close(pipe_fd[1]);
while(isClientWork) {
int epoll_events_count = epoll_wait(epfd, events, 2, -1 );
//处理就绪事件
for(int i = 0; i < epoll_events_count ; ++i){
bzero(&message, BUF_SIZE);
if(events[i].data.fd == socketfd){
//服务端发来消息
int ret = recv(socketfd, message, BUF_SIZE, 0);
if(ret == 0){
printf("服务端关闭了连接,fd=%d\n", socketfd);
close(socketfd);
isClientWork = 0;
exit(0);
}else {
printf("接收消息:%s\n", message);
}
}else if(events[i].data.fd == pipe_fd[0]) {
//子进程写入事件发生,父进程处理并发送服务端
int ret = read(events[i].data.fd, message, BUF_SIZE);
if(ret == 0){
isClientWork = 0;
}else {//将消息发送给服务端
send(socketfd, message, BUF_SIZE, 0);
}
}
}
}
}
if(pid){
//关闭父进程和socket
close(pipe_fd[0]);
close(socketfd);
}else {
//关闭子进程
close(pipe_fd[1]);
}
}
return 0;
}