- 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/
是否可以在蓝牙设备连接到 iOS 或从 iOS 断开连接时收到通知,即使我的应用程序处于后台?在 Android 上,我使用 ACTION_ACL_CONNECTED 和 ACTION_ACL_DIS
我对使用 Delphi 使用 USB 设备感到不舒服,并且对编写设备驱动程序的细节几乎一无所知(尽管我在学习使用 GoASM 进行汇编时遇到过一些)。 该设备可以是 USB 调制解调器或 USB 打印
我正在使用 java/servlet 和前端 JSP 开发 Web 应用程序。 我的页面为用户显示一些信息。 是否有任何适当的方法来检查网络连接是否处于 Activity 状态(开/关)以及支持所有浏
这个问题在这里已经有了答案: C# -- TcpListener.Start() causing SocketException with message "Only one usage of ea
我想向我的应用程序添加功能,以允许在 Windows 操作系统上检测播放/捕获设备的到达和断开连接。就像在 Skype 中一样 - 当您拔下设备时 - 它会显示设备丢失的通知,例如要求您选择另一个。
如何断开 org. apache. http. client.HttpClient 对象? 我正在这样创建它,但它没有任何 disconnect() 或 getHttpConnectionManage
我们正在开发一个 .NET 应用程序,其中一个要求是监视系统是否连接到 Internet。 我们能够获得“以太网电缆断开连接”的 .NET 事件,但如果调制解调器被重置,则不会触发此事件。我不想一直通
我正在尝试与使用自签名证书的后端服务器建立 websocket 连接。在 Firefox 中,我为自签名证书添加了一个异常(exception)。 但是我的 websocket 连接 wss://连接
我正在使用由 Excel 生成的用户窗体来修改 PowerPoint 演示文稿(这是避免需要启用宏的电子表格的迂回方法)。该表单工作得很好,但每次我将焦点放在它上时,Excel 应用程序都会获得焦点(
您好,我成功地开始了视频通话,但是当我断开视频通话时,摄像头 LED 指示灯仍然亮着。我怎样才能完全断开通话? Video.createLocalVideoTrack().then(track =>
您好,我成功地开始了视频通话,但是当我断开视频通话时,摄像头 LED 指示灯仍然亮着。我怎样才能完全断开通话? Video.createLocalVideoTrack().then(track =>
我正在尝试断开客户端与服务器的连接,但服务器仍将其视为已连接。我找不到解决这个问题的方法,关机、断开连接和关闭都不起作用。 我与客户端断开连接并检查服务器的一些代码: 客户: private vo
我正在使用 magic record 来完成我所有的核心数据工作。 一切都很好,除了有时我在后台进行更新时我需要从上下文中分离或断开实体。 例如 ButtonList = [Buttons MR_fi
我正在尝试实现一个颜色选择器,它从屏幕上各处的像素中获取颜色。为此,我计划使用全局鼠标 Hook 来监听 WM_MOUSEMOVE,以便在鼠标四处移动时更新颜色,并监听鼠标点击以确认 (WM_LBUT
我有一个使用 C# 编写的 TLS 连接的客户端/服务器 TCP 场景。 客户端在安静并恢复后无法传递消息(不活动时间约为 25 分钟)。但如果我让客户聊天(每 30 秒),就没有问题。 客户端和服务
我编写的这段代码完美运行,但我担心每 2 秒执行一次 ping 操作会消耗太多资源,或者可能导致互联网连接出现一些问题。 new Thread(() => {
关闭带有附加 MediaPlayer 的 MediaController 的正确方法是什么? 您不能执行 mediaController.setMediaPlayer(null) - 立即调用 upd
从我的笔记本电脑上断开外接显示器后,我丢失了一些应用程序,因为断开连接的显示器仍设置为默认显示器。我的一些窗口试图在断开连接的显示器上显示。 我有一个解决方法,例如右键单击应用程序图标并选择移动,然后
是否可以将分片节点从分片集中取出并独立运行? 例如,每个分片都有一个特定的客户。 谢谢。 最佳答案 是的,这是可能的。 MongoDB“主”节点不知道分片。只有路由器(和配置服务器)知道。如果您使用正
我正在探索 Cassandra 及其复制系统。为此,我在本地创建了 3 个节点并测试了不同的场景。我想暂停节点之间的通信,用相同的键写入不同的值,然后观察它们将决定什么。 那么问题来了:如何限制节点在
我是一名优秀的程序员,十分优秀!