在C套接字中需要多个连接
我编写了以下代码作为连接两个程序的中间代码。在两个不同的系统上运行一个服务器程序和一个客户端程序。该代码将作为这两个程序之间的中间代码在C套接字中需要多个连接,c,linux,sockets,unix,network-programming,C,Linux,Sockets,Unix,Network Programming,我编写了以下代码作为连接两个程序的中间代码。在两个不同的系统上运行一个服务器程序和一个客户端程序。该代码将作为这两个程序之间的中间代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <netdb.h> #include <stdio.h> #include <string.h> #include <stdl
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <netdb.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
//Connect with program on server side
char * serv_con(char app_data[50])
{
int sock, bytes_recieved;
char send_data[1024];
char *recv_data = malloc(1024);
struct hostent *host;
struct sockaddr_in server_addr;
host = gethostbyname("10.47.3.249");
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Socket");
exit(1);
}
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(3128);
server_addr.sin_addr = *((struct in_addr *)host->h_addr);
bzero(&(server_addr.sin_zero),8);
if (connect(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
{
perror("Connect");
exit(1);
}
bytes_recieved=recv(sock,recv_data,1024,0);
recv_data[bytes_recieved] = '\0';
send(sock, app_data, 50, 0);
return recv_data;
//close(sock);
}
//Connect with client app
char * cli_con(char ser_data[50])
{
int sock, connected, bytes_recieved , true = 1;
char send_data [1024];
char *recv_data = malloc(1024);
struct sockaddr_in server_addr,client_addr;
int sin_size;
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Socket");
exit(1);
}
if (setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&true,sizeof(int)) == -1)
{
perror("Setsockopt");
exit(1);
}
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(5000);
server_addr.sin_addr.s_addr = INADDR_ANY;
bzero(&(server_addr.sin_zero),8);
if (bind(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
{
perror("Unable to bind1");
exit(1);
}
if (listen(sock, 5) == -1)
{
perror("Listen");
exit(1);
}
sin_size = sizeof(struct sockaddr_in);
connected = accept(sock, (struct sockaddr *)&client_addr,&sin_size1);
printf("\n I got a connection from (%s , %d)",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
bytes_recieved = recv(connected,recv_data,1024,0);
recv_data[bytes_recieved] = '\0';
send(connected, ser_data,50, 0);
//close(sock);
}
int main()
{
char *ser_data, *app_data;
int pid = fork();
while(1)
{
if(pid == 0)
app_data = serv_con(ser_data);
else
ser_data = cli_con(app_data);
}
}
我应该在代码中做什么修改来纠正此错误?我在linux上工作。提前谢谢
编辑:
我删除了close(sock)
中的注释,并在函数cli\u-con
中添加了close(connect)
。客户端的代码如下所示:
int sock, bytes_recieved;
char send_data[1024],recv_data[1024];
struct hostent *host;
struct sockaddr_in server_addr;
host = gethostbyname("192.168.0.2");
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Socket");
exit(1);
}
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(5555);
server_addr.sin_addr = *((struct in_addr *)host->h_addr);
bzero(&(server_addr.sin_zero),8);
if (connect(sock, (struct sockaddr *)&server_addr,
sizeof(struct sockaddr)) == -1)
{
perror("Connect");
exit(1);
}
while(1)
{
//necessary codes
if (connect(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
{
close(sock);
goto connct;
}
}
但现在运行时,第一个程序不会退出,甚至不会打印
I got a connection from (192.168.0.3 , 45691)
但只需继续运行,不打印任何消息。但另一方面,客户端退出时显示错误:
Unable to bind: Address already in use
I got a connection from (192.168.0.3 , 45691)
Connect: Connection reset by peer
现在该怎么办?这意味着您正在尝试侦听端口5000,但该端口上已经有一个程序正在侦听(可能是您的程序的早期版本,该版本没有正确关闭端口)。请将端口号更改为另一个值,或关闭侦听端口的任何应用程序
由于您使用的是Linux,因此可以使用“netstat-nlp”作为根用户查看哪些程序打开了哪些端口。当客户端断开连接时,您将创建一个新的服务器套接字并将其绑定到同一端口。如果服务器端套接字未关闭,则端口仍在使用中,因此
bind
会失败
通常,套接字程序的服务器端有一个循环accept
,以允许它处理来自多个客户端的连接。这样,bind
和listen
只调用一次
while (connected = accept(sock, (struct sockaddr *)&client_addr,&sin_size1)) {
printf("\n I got a connection from (%s , %d)",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
bytes_recieved = recv(connected,recv_data,1024,0);
recv_data[bytes_recieved] = '\0';
send(connected, ser_data,50, 0);
close(connected);
}
一个提示是,您应该同时关闭侦听套接字和从accept调用获得的套接字
close(connected);
close(sock);
此外,套接字在使用后需要一段时间才能被操作系统释放,因此,它可能会因“地址已在使用”错误而失败。如果代码的SO_REUSEADDR部分执行正确,则可以在代码中再次正确检查
另外,因为您可以尝试在主函数本身中添加侦听套接字创建代码,并将其作为参数传递给cli_con函数。通常遵循的机制是创建一个侦听套接字,并使用它接受来自客户端的多个连接
我希望在您的原始代码中,您对ser_数据和app_数据有适当的内存分配、初始化等。“我从(192.168.0.345691)获得了一个连接”-这也会被打印出来吗?如果代码因为绑定错误而退出,怎么可能呢?我认为这是while循环。它第一次获得连接,第二次进入循环,退出时显示绑定错误。知道应该怎么做吗?我已经用
close(sock)
(在上面的代码中注释)关闭了套接字。但它仍然给出了相同的错误。如何关闭端口?@HarikrishnanT:close()调用会在您使用完套接字后关闭它。错误消息告诉您首先不允许使用套接字,因为其他人正在使用它。为什么要创建新的服务器套接字?套接字关闭后,操作系统会自动释放端口,但这可能需要一些时间。您可以将SO_REUSEADDR选项设置为允许在仍在使用的端口上进行bind
,但通常您要做的是重用套接字,而不是端口。您是否也在cli_con中添加了循环?我认为您的客户机出现了错误,因为您处于一个连续的while循环中,在经过一定数量的连接后,服务器拒绝接受连接。至于服务器端,请在accept之前和之后添加一个打印,以查看是否有任何连接