gpt4 book ai didi

使用 c 中的 select() 函数进行客户端到客户端通信

转载 作者:太空宇宙 更新时间:2023-11-04 03:28:56 25 4
gpt4 key购买 nike

我试图实现客户端与客户端之间的服务器通信。服务器的功能是,当客户端假设客户端 A 向服务器发送消息时,服务器应将该消息转发给另一个客户端,即客户端 B。同样,当客户端 B 向服务器发送消息时,它应该被转发给客户 A。这个程序只涉及两个客户。执行代码时出现的错误是:

Socket Operation on Non-socket

当从客户端 A 收到的消息转发给客户端 B 时出现此错误。我认为问题是由于将客户端 B 的接收地址存储到客户端 A 的地址。我不确定。

My server code so far.

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <signal.h>
#include <unistd.h>
#include <errno.h>
#include <sys/time.h>
#define SERVER_PORT 5009


int main(){
unsigned int sockfd, c,c1,c2, clientlen, clientfd;
struct sockaddr_in server;
struct sockaddr_in client1;
int clientsocks[2];
char rmsg1[100], msg1[100],rmsg2[100], msg2[100];
char w_msg[] = "Connection to server established";

fd_set readfds; // For temp file descriptor list.

clientsocks[0] = 0 ;
clientsocks[1] = 0 ;
//Socket Creation Process.
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if( sockfd < 0){
perror("Socket cannot be created");
}

//For reusing the socket.
int on = 1;
if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)
//Socket address
bzero((char *) &server, sizeof(server));
server.sin_family = AF_INET; // IPv4 internet Protocols
inet_aton("127.0.0.1", &server.sin_addr);
server.sin_port = htons(SERVER_PORT);

//Binding socket to address.
if (bind( sockfd, (struct sockaddr*)&server, sizeof (server) ) < 0 ){
perror("Bind Error");
exit(EXIT_FAILURE);
}

//Listen to accept connection.
if( listen(sockfd, SOMAXCONN) < 0 ){
perror("Error in Listen");
exit(EXIT_FAILURE);
}

unsigned int new_sock;
clientlen =sizeof(client1);
int activity;
while(1){

//Clear socket set.
FD_ZERO(&readfds);

//Adding main sockfd to the socket set.
FD_SET(sockfd, &readfds);
unsigned int max_sd = sockfd;

//Add child sockets to set.
for(int i=0 ; i<2; i++){
c = clientsocks[i];
if(c > 0)
FD_SET(c, &readfds);
if(c > max_sd)
max_sd = c;
}

activity = select(max_sd + 1, &readfds, NULL, NULL, NULL);
if(activity < 0){
perror("Error in select()");
exit(EXIT_FAILURE);
}

//Incoming connection when something happens on sockfd.
if( FD_ISSET(sockfd, &readfds)){
new_sock = accept(sockfd, (struct sockaddr *) &client1, &clientlen);
if(new_sock > 0){
for(int i=0; i<2; i++){
if(clientsocks[i] == 0){
clientsocks[i] = new_sock;
break;
}
}
}
if( new_sock < 0){
perror("Error Accepting");
exit(EXIT_FAILURE);
}
if( send(new_sock, w_msg, strlen(w_msg), 0) != strlen(w_msg)){
perror("Welcome message");
exit(EXIT_FAILURE);
}
c1 = clientsocks[0];
c2 = clientsocks[1];
FD_SET(c1, &readfds);
FD_SET(c2, &readfds);
}

//Else if its not a new incoming connection.
if(FD_ISSET(c1, &readfds)){
if(recv(c1, rmsg1, 100, 0) < 0){
perror("Receive 1");
exit(EXIT_FAILURE);
}
printf("Client1 >> %s\n", rmsg1);
//Forwarding to Client B.
if( send(c2, rmsg1, 100, 0) < 0){
perror("Error forwarding to 2");
exit(EXIT_FAILURE);
}
}
if(FD_ISSET(c2, &readfds)){
if(recv(c2, rmsg2, 100, 0) < 0){
perror("Receive 2");
exit(EXIT_FAILURE);
}
printf("Client2 >> %s\n", rmsg2);
if( send(c1, rmsg2, 100, 0) < 0 ){
perror("Error Forwarding to 1");
exit(EXIT_FAILURE);
}
}
}
close(sockfd);
return 1;
}

我的问题只涉及两个客户。如果您也能指出其他一些改进,我将不胜感激。

最佳答案

您遇到了一些结构性问题,这并不是 StackOverflow 成员真正需要解决的问题。

  1. 决定每个客户端是要连接到同一个套接字还是他们自己唯一的套接字,相应地听。
  2. 如果你有一个共享连接,这似乎是你想要的......我所说的共享连接是指你在一个端口上监听并进行多次接受,那么在一个给定的接受之后你不能假设谁连接了直到你从端口读取一些数据。您应该让每个客户端发送一些信息,以便您知道是 A 连接然后 B 还是 B 连接然后 A。
  3. 您的 setsockopt 周围的代码似乎您忘记了{}
  4. 保存客户端套接字时,您可以使用来自客户端的信息来确定要使用的数组槽。也许您有 100 个客户端,它们每个都连接,然后发送一个 32 位字(其中包含它们的客户端编号,在 1-100 之间)。
  5. 一旦您可以读取来自不同客户端的消息(可能想暂时将它们打印出来,以便您了解发生了什么),您就可以构建各种消息。目标是客户端 A 能够向服务器询问客户端 B 的联系信息,以便 A 可以直接连接到 B 并向 B 发送消息。
  6. 或者,客户端 26 应该能够向服务器发送一条消息,而不是 #5,表明它是给客户端 45 的,服务​​器应该能够检查已 checkin 的客户端数组,然后发送消息到阵列槽 45 中的客户端。
  7. 客户端需要具有唯一的 ID/编号,这样服务器才能将客户端 ID 映射到客户端数组套接字。
  8. 其中一些错误直接处理比导致服务器退出更好。也许您可以关闭与导致错误的套接字索引关联的套接字。您的错误消息应指出导致错误的客户端/套接字索引。通常,您需要更多调试消息和消息中的更多信息。

关于使用 c 中的 select() 函数进行客户端到客户端通信,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38733264/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com