- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我使用 pthread 编写了一个相当简单的 C 聊天服务器。服务器工作正常,客户端使用telnet连接到服务器,然后客户端可以相互通信。每个客户端都在自己的线程中启动。当任何客户端决定断开连接时,问题就会出现。即使我关闭客户端套接字并完成其线程(它确实如此,或者至少 gdb 是这么说的),这种干扰以某种方式关闭了服务器端的所有通信,即使其他线程仍在运行。此后,客户端无法相互通信,因此服务器变得毫无用处。我不知道如何解决这个问题,我请求你的帮助。谢谢。
这是我的代码。我包含完整的代码,因为我不确定我的错误在哪里
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <pthread.h>
int user[30]; // sockets of connected users
char names[30][20];
int user_count=0; // number of active connections
int listen_sock; // main socket, recieves connections
char buffer[500];
pthread_t *thready[30];
pthread_mutex_t mutex;
struct arg_struct {
int arg1;
int arg2;
};
void sending(char *text, int i) {
char buffer2[504];
strcpy(buffer2, "\r\0");
strcat(buffer2, text);
strcat(buffer2, "\r\0");
write(user[i], buffer2, strlen(buffer2));
}
void welcome(int id) {
int len;
int sock;
char name[20];
pthread_mutex_lock(&mutex);
sock = user[id];
write(sock,"\rEnter your name: ", 18);
// Read returns number of characters recieved
len = read(sock,name,20);
// Add zero to the end of string (read doesnt do it)
name[len]='\0';
while (name[len] < 32)
{
name[len]='\0';
len--;
}
// snprintf(buffer,"Welcome %s!\n",name);
write(sock, "Welcome\r\n", 7);
strcpy(names[id], name);
sending(buffer,id);
pthread_mutex_unlock(&mutex);
}
void echo(char *text) {
int i;
for (i=0; i<user_count; i++) {
sending(text, i);
}
}
void disconnect(int id) {
int i;
char deadname[20];
strcpy(deadname,names[id]);
sending("You have been disconnected\n",id);
shutdown(user[id], SHUT_RDWR);
close(user[id]);
for (i=id; i<user_count-1; i++) {
user[i]=user[i+1];
strcpy(names[i],names[i+1]);
}
user_count--;
sprintf(buffer,"%s has disconnected\n",deadname);
echo(buffer);
}
void finish() {
echo("Shuting down, disconnecting everyone!\n");
while (user_count > 0) disconnect(user_count-1);
shutdown(listen_sock, SHUT_RDWR);
close(listen_sock);
pthread_mutex_destroy(&mutex);
exit(0);
}
void getcomm(char *inpstr, char *comm) { /* First letter from inpstr goes to comm,
and is removed from inpstr */
int wpos=0,leng; /* Splits first word from the rest of the sentence */
char *zal;
zal=inpstr;
leng=strlen(zal);
while (*inpstr>32 && wpos<14) {
*comm=*inpstr++;
comm++;
wpos++;
}
inpstr=(char *)memmove(zal,inpstr+1,leng);
*comm='\0';
}
void input(int id) {
}
void* communication(void* par){
int sock = ((int*)par)[0];
int id = ((int*)par)[1];
free((int*)par);
welcome(id);
while(1){
char line[500];
char command[20];
char name2[16];
int len,sock;
// 1. read line from user
len=read(user[id],line,499);
pthread_mutex_lock(&mutex);
line[len]='\0';
if (len == 0) { // User has terminated the connection
continue;
}
while (line[len] < 32) { // Remove special chars
line[len]='\0';
len--;
}
// 2. Get first word
getcomm(line,command);
// 3. Recognise command and execute it
len=strlen(command);
if (strncmp(command,"quit",4) == 0) {
disconnect(id);
pthread_exit(NULL);
}
else if (!strncmp(command,"tell",len)) {
getcomm(line,name2);
sprintf(buffer,"%s tells %s: %s\n", names[id], name2, line);
echo(buffer);
}
else if (!strncmp(command,"shut",len)) {
finish();
}
// 4. Unknown command
else {
sprintf(buffer,"%s tells %s %s\n", names[id], command, line);
echo(buffer);
}
pthread_mutex_unlock(&mutex);
}
}
int main( int argc, char *argv[] ) {
int pid;
struct sockaddr_in bind_addr;
struct sockaddr_in acc_addr;
int size, user_id, i, sel,on;
int port_number;
char * eptr = NULL;
fd_set readmask;
pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutex_init(&mutex, NULL);
listen_sock = socket(AF_INET, SOCK_STREAM, 0);
if (listen_sock==-1) {
perror("socket()");
pthread_mutex_destroy(&mutex);
exit(-1);
}
on = 1;
if (argc == 1){
perror("Please specify port number");
pthread_mutex_destroy(&mutex);
//exit(0);
}
if (argc == 2){
port_number = (int) strtol(argv[1], &eptr, 10);
if (*eptr != '\0'){
perror("Invalid Port Number!");
pthread_mutex_destroy(&mutex);
exit(-1);
}
}
port_number = 7501;
bind_addr.sin_family = AF_INET;
bind_addr.sin_addr.s_addr = INADDR_ANY;
bind_addr.sin_port = htons(port_number);
size=sizeof(struct sockaddr_in);
setsockopt(listen_sock, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on));
if (bind(listen_sock, (struct sockaddr *)&bind_addr, size)==-1) {
perror("bind()");
pthread_mutex_destroy(&mutex);
exit(-1);
}
listen(listen_sock, 10);
while(1) {
FD_ZERO(&readmask);
FD_SET(listen_sock, &readmask);
if (FD_ISSET(listen_sock, &readmask)) {
user_id = user_count++;
int * soc = (int*)malloc(2*sizeof(int));
soc[0] = accept(listen_sock,(struct sockaddr *)&acc_addr, &size);
soc[1] = user_id;
pthread_mutex_lock(&mutex);
user[user_id] = soc[0];
pthread_mutex_unlock(&mutex);
if( ( pthread_create( &thready[user_id], NULL, communication, (void*)soc ) ) < 0)
{
perror("could not create thread");
return 1;
}
echo(buffer);
continue;
}
}
pthread_mutex_destroy(&mutex);
return 0;
}
最佳答案
在调用 pthread_exit() 之前,您似乎没有解锁互斥锁。这应该会阻止所有其他通信线程。
此外,您是否错过了 FD_SET() 之后的 select() 调用?
关于当客户端断开连接时,C 服务器停止响应,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14344162/
我知道这个问题可能已经被问过,但我检查了所有这些,我认为我的情况有所不同(请友善)。所以我有两个数据集,第一个是测试数据集,第二个是我保存在数据框中的预测(预测值,这就是没有数据列的原因)。我想合并两
在 .loc 方法的帮助下,我根据同一数据框中另一列中的值来识别 Panda 数据框中某一列中的值。 下面给出了代码片段供您引用: var1 = output_df['Player'].loc[out
当我在 Windows 中使用 WinSCP 通过 Ubuntu 连接到 VMware 时,它提示: The server rejected SFTP connection, but it lis
我正在开发一个使用 xml web 服务的 android 应用程序。在 wi-fi 网络中连接时工作正常,但在 3G 网络中连接时失败(未找到 http 404)。 这不仅仅发生在设备中。为了进行测
我有一个XIB包含我的控件的文件,加载到 Interface Builder(Snow Leopard 上的 Xcode 4.0.2)中。 文件的所有者被设置为 someClassController
我在本地计算机上管理 MySQL 数据库,并通过运行以下程序通过 C 连接到它: #include #include #include int main(int argc, char** arg
我不知道为什么每次有人访问我网站上的页面时,都会打开一个与数据库的新连接。最终我到达了大约 300 并收到错误并且页面不再加载。我认为它应该工作的方式是,我将 maxIdle 设置为 30,这意味着
希望清理 NMEA GPS 中的 .txt 文件。我当前的代码如下。 deletes = ['$GPGGA', '$GPGSA', '$GPGSV', '$PSRF156', ] searchquer
我有一个 URL、一个用户名和一个密码。我想在 C# .Net WinForms 中建立 VPN 连接。 你能告诉我从哪里开始吗?任何第三方 API? 代码示例将受到高度赞赏... 最佳答案 您可以像
有没有更好的方法将字符串 vector 转换为字符 vector ,字符串之间的终止符为零。 因此,如果我有一个包含以下字符串的 vector "test","my","string",那么我想接收一
我正在编写一个库,它不断检查 android 设备的连接,并在设备连接、断开连接或互联网连接变慢时给出回调。 https://github.com/muddassir235/connection_ch
我的操作系统:Centos 7 + CLOUDLINUX 7.7当我尝试从服务器登录Mysql时 [root@server3 ~]# Mysql -u root -h localhost -P 330
我收到错误:Puma 发现此错误:无法打开到本地主机的 TCP 连接:9200(连接被拒绝 - 连接(2)用于“本地主机”端口 9200)(Faraday::ConnectionFailed)在我的
请给我一些解决以下错误的方法。 这是一个聊天应用....代码和错误如下:: conversations_controller.rb def create if Conversation.bet
我想将两个单元格中的数据连接到一个单元格中。我还想只组合那些具有相同 ID 的单元格。 任务 ID 名称 4355.2 参与者 4355.2 领袖 4462.1 在线 4462.1 快速 4597.1
我经常需要连接 TSQL 中的字段... 使用“+”运算符时 TSQL 强制您处理的两个问题是 Data Type Precedence和 NULL 值。 使用数据类型优先级,问题是转换错误。 1)
有没有在 iPad 或 iPhone 应用程序中使用 Facebook 连接。 这个想法是登录这个应用程序,然后能够看到我的哪些 facebook 用户也在使用该应用程序及其功能。 最佳答案 是的。
我在连接或打印字符串时遇到了一个奇怪的问题。我有一个 char * ,可以将其设置为字符串文字的几个值之一。 char *myStrLiteral = NULL; ... if(blah) myS
对于以下数据 - let $x := "Yahooooo !!!! Select one number - " let $y := 1 2 3 4 5 6 7 我想得到
我正在看 UDEMY for perl 的培训视频,但是视频不清晰,看起来有错误。 培训展示了如何使用以下示例连接 2 个字符串: #!usr/bin/perl print $str = "Hi";
我是一名优秀的程序员,十分优秀!