我正在使用套接字开发一个简单的聊天应用。它是客户端 - 服务器驱动的。 一切都很完美,除了这个我无法找到的错误:如果我是第一个连接到服务器的客户端,我收到的每条消息都只在屏幕上打印一次(因为它应该是)。然而,如果我是连接的第二个客户端,我收到的每条消息都打印两次,第三个客户端看到他的消息三次,依此类推。我真的因为这个而疯了:
当我通过套接字收到东西时,我打印出来就像这样:
printf (" >%s \n", received);
因此,如果我收到消息hello,由用户发送“charles”我打印消息“> charles:hello”,令我惊讶的是重复打印没有“>”我添加的字符(副本将是“charles:hello”),因此无法找到正在进行打印的位置(上面的printf()
是我代码中唯一的printf()
。
我会感激任何帮助。以下是从socket接收的代码部分:
while (1) {
recv(newsockfd, recibido, 255, 0);
printf (">%s \n",recibido);
}
此代码由线程执行。
谢谢,伙计们。
以下是客户的完整代码:
void *Atender(void *estructura){
DATOS *info;
info = (DATOS *)estructura;
int newsockfd=info->socket;
char recibido[255];
memset(recibido,1,255);
while (1){
recv(newsockfd, recibido, 255, 0);
printf (">%s \n",recibido);
memset(recibido,0,255);
}
}
main(int argc, char *argv[]) {
int sockfd;
struct sockaddr_in serveraddr;
char *server;
Hilos = malloc((200)*sizeof(pthread_t));
/* Remember the program name for error messages. */
programname = argv[0];
//Realizo lectura e imprimo para chequear que se leyo todo bien.
lectura(argv,argc);
/* Who's the server? */
server=host;
/* Get the address of the server. */
bzero(&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET;
serveraddr.sin_addr.s_addr = inet_addr(server);
serveraddr.sin_port = htons(atoi(puerto));
/* Open a socket. */
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
fatalerror("can't open socket");
/* Connect to the server. */
if (connect(sockfd, (struct sockaddr *) &serveraddr,
sizeof(serveraddr)) < 0)
fatalerror("can't connect to server");
enviarInfoInicial(sockfd,nombre,"Actual");
char buff[200];
memset(buff,1,200);
//Creacion del hilo que se quedara siempre escuchando
DATOS d;
d.socket=sockfd;
pthread_create(&Hilos[iterador_hilos],NULL,Atender,(void *)&d);
iterador_hilos++;
//El que no es un hilo se queda siempre esperando por si se quiere
//introducir un comando
while (1) {
fgets(buff, sizeof buff, stdin);
mandarInstruccion(sockfd,buff);
if ((buff[0] == 'f') & (buff[1] == 'u') & (buff[2] == 'e')){
printf("Cchat se ha cerrado. \n");
close(sockfd);
exit(0);
}
memset(buff,1,200);
}
}
代码是西班牙语,但总体思路是可以理解的。 DATOS
结构只是我传递给创建的线程的结构,并且只包含该线程将从中读取的套接字(先前发布的while(1)
)。 “ATENDER”是线程将执行的过程。
以下是服务器的相关代码:
void EjecutarInstruccion(char tipo[],char argumento[],char usuario[]) {
if (!strcmp(tipo,"men")) {
char sala[20];
char total[300];
memset(total,0,300);
strcpy(sala,ObtenerSalaUsuario(usuario));
int i=0;
for (i;i<200;i++) {
if (!strcmp((lusuarios[i].sala),sala)) {
strcat(total,usuario);
strcat(total,": ");
strcat(total,argumento);
send (lusuarios[i].socket, total, strlen(total)+1, 0 );
}
}
}
}
在服务器端我有一个用户数组,每个用户都包含他的用户名,他所在的聊天室和他的套接字(fd)。当服务器需要发送消息时,我会检查用户数组,并且只有当他在同一个聊天室时才将消息发送给该用户。 感谢。
答案 0 :(得分:2)
您不检查返回的字节数recv
。它不一定会读取所有未完成的数据。您可能需要在循环中读取,直到收到所有字节。