-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathContact_Server.cpp
320 lines (296 loc) · 9.8 KB
/
Contact_Server.cpp
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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
#include <windows.h>
#include <winsock2.h>
#include <condition_variable>
#include <iostream>
#include <mutex>
#include <queue>
#include <thread>
#include "server_detail.h"
#pragma comment(lib, "ws2_32.lib")
using namespace std;
typedef struct _CLIENT_ {
int status; // current status
SOCKET socket; // current user socket
char username[NAME_LEN + 1]; // user name
int userId; // user id
} Client;
Client client[MAX_CLIENT_NUM] = {0};
queue<string> message_q[MAX_CLIENT_NUM];
int current_client_num = 0;
// thread
DWORD send_thread[MAX_CLIENT_NUM];
DWORD chat_thread[MAX_CLIENT_NUM];
std::mutex current_client_num_mtx; //用户数量互斥量
std::mutex mtx; //定义一个mutex对象,互斥量
HANDLE client_mutex[MAX_CLIENT_NUM] = {0}; //用户数组每个用户的互斥量
HANDLE client_semaphore[MAX_CLIENT_NUM] = {0}; //定义用户信号量
/***
* CLASS SERVER FUNCTION
*
*/
Server::Server() {
cout << "服务端启动中" << endl;
winsock_ver = MAKEWORD(2, 2);
addr_len = sizeof(SOCKADDR_IN);
addr_svr.sin_family = AF_INET;
addr_svr.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
addr_svr.sin_port = htons(SERVER_PORT);
memset(buf_ip, 0, IP_BUF_SIZE);
//
ret_val = ::WSAStartup(winsock_ver, &wsa_data);
if (ret_val != 0) {
cerr << "WSA failed to start up! Error code" << ::WSAGetLastError() << endl;
system("pause");
exit(1);
}
//
sock_svr = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (sock_svr == INVALID_SOCKET) {
cerr << "Failed to create server socket!Error code: " << ::WSAGetLastError()
<< endl;
::WSACleanup();
system("pause");
exit(1);
}
cout << "Server socket is created successfully!" << endl;
//
if ((ret_val = ::bind(sock_svr, (SOCKADDR*)&addr_svr, addr_len)) != 0) {
//绑定失败
cerr << "Failed to bind server socket!Error code: " << ::WSAGetLastError()
<< endl;
::WSACleanup();
system("pause");
exit(1);
}
cout << "Server socket is bound successfully...\n";
//
if ((ret_val = ::listen(sock_svr, 20)) == SOCKET_ERROR) {
cerr << "Server socket failed to listen!Error code: " << ::WSAGetLastError()
<< "\n";
::WSACleanup();
system("pause");
exit(1);
}
cout << "Server socket started to listen...\n";
//
cout << "服务端启动成功!" << endl;
}
Server::~Server() {
::closesocket(sock_svr);
::closesocket(sock_clt);
::WSACleanup();
cout << "服务端关闭" << endl;
}
void* WINAPI ChatClientThrea(LPVOID client_info);
void Server::WaitClient() {
int sum = 0;
while (true) {
//等待客户端连接然后创建新线程处理客户端
SOCKET client_socket = ::accept(sock_svr, NULL, NULL);
if (client_socket == -1) {
perror("accept");
exit(1);
}
if (::current_client_num >= MAX_CLIENT_NUM) {
//连接数量太多
if (send(client_socket, "ERROR", strlen("ERROR"), 0) < 0) {
perror("send");
::shutdown(client_socket, 2);
continue;
}
} else {
if (send(client_socket, "OK", strlen("OK"), 0) < 0) {
cout<<WSAGetLastError()<<endl;
perror("ok");
}
}
//与客户端连接成功
//获取用户名
char client_name[NAME_LEN + 1] = {0};
int state = recv(client_socket, client_name, NAME_LEN, 0);
if (state < 0) {
perror("recv");
shutdown(client_socket, 2);
continue;
} else if (state == 0) {
shutdown(client_socket, 2);
continue;
}
for (int i = 0; i < MAX_CLIENT_NUM; i++) {
if (!client[i].status) {
//找到空的位置,然后将该用户放进去
::current_client_num_mtx.lock();
//加锁
memset(client[i].username, 0, NAME_LEN);
strcpy(client[i].username, client_name);
client[i].userId = i;
client[i].status = 1;
client[i].socket = client_socket;
//用户信息填充完毕,为该用户创建互斥量
::client_mutex[i] = ::CreateMutex(NULL, FALSE, NULL);
::current_client_num++;
//为用户创建信号量,初始化为1
::client_semaphore[i] = ::CreateSemaphore(NULL, 0, BUFFER_LEN, NULL);
::current_client_num_mtx.unlock();
CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ChatClientThrea,
(void*)&client[i], 0, &::chat_thread[i]);
std::cout << client[i].username << " "
<< "join in the chat room. Online User Number:"
<< current_client_num << endl;
break;
}
}
}
for (int i = 0; i < MAX_CLIENT_NUM; i++)
if (client[i].status) shutdown(client[i].socket, 2);
shutdown(sock_svr, 2);
}
/**
* 消息发送函数
*
*/
void* WINAPI SendThrea(LPVOID client_info) {
Client* client_detail = (struct _CLIENT_*)client_info;
while (true) {
WaitForSingleObject(&::client_mutex[client_detail->userId], INFINITE);
while (::message_q[client_detail->userId].empty()) {
//当消息队列为空
::WaitForSingleObject(&::client_semaphore[client_detail->userId],
INFINITE);
// ::WaitForSingleObject(&::client_mutex[client_detail->userId],INFINITE);
}
//发送消息
while (!::message_q[client_detail->userId].empty()) {
string message_buffer = message_q[client_detail->userId].front();
int size = message_buffer.length();
int trans_len = BUFFER_LEN > size ? size : BUFFER_LEN;
while (size > 0) {
int retlen =
::send(client_detail->socket, message_buffer.c_str(), trans_len, 0);
if (retlen < 0) {
perror("send");
return NULL;
}
size -= retlen;
message_buffer.erase(0, retlen);
trans_len = BUFFER_LEN > size ? size : BUFFER_LEN;
}
message_buffer.clear();
message_q[client_detail->userId].pop();
}
//释放互斥量
ReleaseMutex(&::client_mutex[client_detail->userId]);
}
}
/**
* 消息接收线程
*/
void* WINAPI RecvMessage(LPVOID client_info) {
Client* client_detail = (struct _CLIENT_*)client_info;
// message
string message_buffer;
int message_len = 0;
// buffer
char buffer[BUFFER_LEN + 1];
int buffer_len = 0;
// recvive message
while ((buffer_len = recv(client_detail->socket, buffer, BUFFER_LEN, 0)) >
0) {
//接收到消息
for (int i = 0; i < buffer_len; i++) {
// the start of a new message
if (message_len == 0) {
char temp[100];
sprintf(temp, "%s:", client_detail->username);
message_buffer = temp;
message_len = message_buffer.length();
}
message_buffer += buffer[i];
message_len++;
if (buffer[i] == '\n') {
// sned to every client
for (int j = 0; j < MAX_CLIENT_NUM; j++) {
if (::client[j].status &&
::client[j].socket != client_detail->socket) {
WaitForSingleObject(&::client_mutex[j], INFINITE);
::message_q[j].push(message_buffer);
//信号量
::ReleaseMutex(&::client_semaphore[j]);
::ReleaseMutex(&::client_mutex[j]);
}
}
message_buffer.clear();
message_len = 0;
}
}
buffer_len = 0;
::memset(buffer, 0, sizeof(buffer));
}
return NULL;
}
/**
* 聊天线程处理
*/
void* ChatClientThrea(LPVOID client_info) {
::Client* client_detail = (struct _CLIENT_*)client_info;
// send welcome message
char welcome[100];
sprintf(welcome,
"Hello %s, Welcome to join the chat room. Online User Number: %d\n",
client_detail->username, current_client_num);
::WaitForSingleObject(&client_mutex[client_detail->userId], INFINITE);
::message_q->push(welcome);
// signal a semaphore
::ReleaseSemaphore(&::client_semaphore[client_detail->userId], 1, NULL);
::ReleaseMutex(&client_mutex[client_detail->userId]);
memset(welcome, 0, sizeof(welcome));
sprintf(welcome, "New User %s join in! Online User Number: %d\n",
client_detail->username, current_client_num);
// send messages to other users
for (int j = 0; j < MAX_CLIENT_NUM; j++) {
if (::client[j].status && client[j].socket != client_detail->socket) {
::WaitForSingleObject(&client_mutex[j], INFINITE);
message_q[j].push(welcome);
::ReleaseSemaphore(&::client_semaphore[j], 1, NULL);
::ReleaseMutex(&client_mutex[j]);
}
}
//创建发送线程
CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)SendThrea,
(LPVOID)client_detail, 0,
&::send_thread[client_detail->userId]);
//接收消息
::RecvMessage(client_info);
//因为recv()函数为阻塞运行模式,当接收消息函数返回时,那么这个用户已经下线
//加锁
::current_client_num_mtx.lock();
client_detail->status = 0;
::current_client_num--;
::current_client_num_mtx.unlock();
//显示用户离开的消息
std::cout << client_detail->username
<< "left the chat room.Online Person Number:"
<< ::current_client_num << endl;
char bye[100];
sprintf(bye, "%s left the chat room. Online Person Number: %d\n",
client_detail->username, current_client_num);
//发送消息给其他人告诉该用户已经离线
for (int j = 0; j < MAX_CLIENT_NUM; j++) {
if (::client[j].status && ::client[j].socket != client_detail->socket) {
::WaitForSingleObject(&::client_mutex[j], INFINITE);
message_q[j].push(bye);
::ReleaseSemaphore(&::client_semaphore[j], 1, NULL);
::ReleaseMutex(&::client_mutex[j]);
}
}
::CloseHandle(::client_mutex[client_detail->userId]);
::CloseHandle(::client_semaphore[client_detail->userId]);
::TerminateThread(&send_thread[client_detail->userId], 0);
// pthread_cancel(send_thread[client_detail->userId]);
return NULL;
}
int main(void) {
system("chcp 65001");
Server server = Server();
server.WaitClient();
return 0; }