C 套接字:客户端之间的 connect() 和/或 accept() 存在问题。 111:连接被拒绝
Posted
技术标签:
【中文标题】C 套接字:客户端之间的 connect() 和/或 accept() 存在问题。 111:连接被拒绝【英文标题】:C socket: problem with connect() and/or accept() between clients. 111: Connection refused 【发布时间】:2010-06-02 18:40:55 【问题描述】:女士们先生们,大家好,accept()
有点问题。我有多个客户端和一台服务器。客户端可以与服务器正常连接和通信。但在某一时刻,我需要一些客户端直接相互连接,我在那里遇到了一些困难。客户端有一堆线程在运行,其中一个是handle_connection()
,它有一个while(1)
,永远循环到listen()
和accept()
任何传入的连接。每当一个客户端尝试connect()
到其他客户端时,connect()
返回一个错误,111:连接被拒绝。我知道我有正确的 IP 地址和正确的端口(我已经为客户端之间的连接指定了一个端口)。等待连接的客户端没有注意到任何东西,没有新的连接,nada。我复制了部分代码,希望有人能指出我做错了什么。感谢您的任何意见!
这是所有客户端代码。
void * handle_connections(void * arg)
是一个永远循环到accept()
任何传入连接的线程。我的服务器发生了非常相似的情况,并且运行良好。 (不知道为什么它在这里不起作用..)这是等待新传入连接的客户端部分。
int handle_request(void * arg, struct message * msg)
在程序中的某一时刻被调用,并尝试连接到在 struct message * msg
中指定的客户端,其中包括带有 IP 地址和端口号等的 struct sockaddr_in
。
#define SERVER_PORT 10000
#define CLIENT_PORT 3456
#define MAX_CONNECTION 20
#define MAX_MSG 50
void * handle_connections(void * arg)
struct fd_info * info;
struct sockaddr_in client_address;
struct timeval timeout;
fd_set readset, copyset;
bzero((char * ) &client_address, sizeof(client_address)); // copy zeroes into string
client_address.sin_family = AF_INET;
client_address.sin_addr.s_addr = htonl(INADDR_ANY);
client_address.sin_port = htons(CLIENT_PORT);
sockfd = socket(AF_INET, SOCK_STREAM, 0);
rv = listen(sockfd,MAX_CONNECTION);
while(1)
new_sockfd = accept(sockfd, (struct sockaddr *) &client_address, &client_addr_len); //blocks
if (new_sockfd < 0)
printf("C: ERROR accept() %i: %s \n", errno, strerror(errno));
sleep(2);
else
printf("C: accepted\n");
FD_SET(new_sockfd, &readset); // sets bit for new_sockfd to list of sockets to watch out for
if (maxfd < new_sockfd)
maxfd = new_sockfd;
if (minfd > new_sockfd)
minfd = new_sockfd;
//end if else (new_sockfd)
// end of the forever while loop
int handle_request(void * arg, struct message * msg)
char * cname, gname, payload;
char * command[3];
int i, rv, sockfd, client_addr_len;
struct sockaddr_in client_address;
struct fd_info * info;
info = (struct fd_info *) arg;
sockfd = info->sock_fd;
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd == -1)
printf("HR: ERROR socket() %i: %s \n", errno, strerror(errno));
break;
else if (sockfd > 0)
printf("HR: new socks is %i \n", sockfd);
printf("HR: sin_family is %i: %i\n", msg->peer.client_address.sin_family, msg->peer.client_address.sin_port);
//*************************************************************
//this is the part that returns error 111: Connection refused!!!
//*************************************************************
rv = connect(sockfd, (struct sockaddr *) &msg->peer.client_address, sizeof(struct sockaddr));
if (rv == -1)
printf("HR: ERROR: connect() %i: %s \n", errno, strerror(errno));
printf("HR: at %li \n", msg->peer.client_address.sin_addr.s_addr);
break;
else if (rv > 0)
info->max_fd = sockfd;
printf("HR: connected successfully!! \n");
【问题讨论】:
您的客户端是否在防火墙后面? 家用网络路由器如今越来越普遍。大多数路由器会阻止未在路由器设置中明确配置的入站流量通过。即使不使用路由器,个人防火墙也并不少见。无论哪种方式,如果客户端位于 NAT/防火墙后面,则该客户端通常需要连接到另一个客户端,因此 NAT/防火墙看到的是出站连接而不是入站连接。如果两个客户端都在 NAT/防火墙后面,那么您需要通过您的服务器代理数据,以便两个客户端都连接出站。 好吧,客户端和服务器都在同一个网络中,但客户端可以很好地连接到服务器。所以我认为防火墙不会成为问题。 【参考方案1】:您的问题是您没有在listen()
之前调用bind()
,因此您正在侦听随机选择的端口上的连接,而不是您要侦听的端口。
你需要:
if (bind(sockfd, &client_address, sizeof client_address)))
perror("bind");
/* Handle this error somehow */
listen()
之前。
请注意,传递给accept()
的sockaddr
结构不是要监听的地址——它是accept()
的输出参数,用于存储accept()
返回时已连接的客户端的地址。
【讨论】:
再次感谢,伙计。我真的不认为我直到现在才理解 bind 做了什么,哈哈......以上是关于C 套接字:客户端之间的 connect() 和/或 accept() 存在问题。 111:连接被拒绝的主要内容,如果未能解决你的问题,请参考以下文章
C++ 套接字:当客户端调用 connect() 时,accept() 挂起,但 accept() 响应 HTTP GET 请求