我知道这个问题看起来很典型,而且已经回答了很多次,但我认为如果你阅读了详细信息,它就不那么常见了(我没有找到)。
重点是,我正在c中开发一个unix服务,它打开一个套接字并等待连接,当我有连接时,我创建一个新的进程来处理它,这样就可以同时打开多个连接。
int newfd = accept(sockfd, (struct sockaddr *)&clientaddr, (socklen_t*)&clientaddr_size);
稍后(在一些其他方法和代码之后和内部)子进程将连接信息保存到BBDD,我还需要在那个精确的时刻获得打开正在处理的连接的IP地址。
由于可以同时有多个连接,并且我传递给accept方法的变量struct sockaddr_in clientaddr
对于所有进程都是共享的我不确定以后用这种方式获取IP地址信息是否是个好主意,因为这样我就可以从另一个打开的连接中获取IP地址。
我希望能够从accept方法(返回的整数)获得的文件描述符int newfd
访问IP地址。有可能吗?或者我误解了文件描述符函数?
好的。多亏了@alk和@rileyberton,我找到了正确的方法,getpeername:
int sockfd;
void main(void) {
//[...]
struct sockaddr_in clientaddr;
socklen_t clientaddr_size = sizeof(clientaddr);
int newfd = accept(sockfd, (struct sockaddr *)&clientaddr, &clientaddr_size);
//fork() and other code
foo(newfd);
//[...]
}
void foo(int newfd) {
//[...]
struct sockaddr_in addr;
socklen_t addr_size = sizeof(struct sockaddr_in);
int res = getpeername(newfd, (struct sockaddr *)&addr, &addr_size);
char *clientip = new char[20];
strcpy(clientip, inet_ntoa(addr.sin_addr));
//[...]
}
因此,现在在另一个过程中,我可以获得发起连接的客户端的IP地址(在"字符串"clientip
中),该客户端仅携带通过accept方法获得的文件描述符newfd
。
您将使用getsockname()
(http://linux.die.net/man/2/getsockname)以获取绑定套接字的IP。
之前也回答过,这里:C-来自文件描述符的公共IP
Statefull连接由两个端点唯一标识Peer(地址:端口)<=>我的(地址:港口)。获取此信息需要getpeername()和getsockname()。
找到了这个有用的示例
我就是这样做的
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <time.h>
int main(int argc, char *argv[])
{
int listenfd = 0, connfd = 0;
struct sockaddr_in serv_addr;
socklen_t len;
char sendBuff[1025];
time_t ticks;
listenfd = socket(AF_INET, SOCK_STREAM, 0);
memset(&serv_addr, '0', sizeof(serv_addr));
memset(sendBuff, '0', sizeof(sendBuff));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
serv_addr.sin_port = htons(5000);
bind(listenfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr));
listen(listenfd, 10);
while(1)
{
connfd = accept(listenfd, (struct sockaddr*)NULL, NULL);
ticks = time(NULL);
snprintf(sendBuff, sizeof(sendBuff), "%.24srn", ctime(&ticks));
printf("%d n",connfd);
getpeername(connfd, (struct sockaddr*)&serv_addr, &len );
printf("Peer connected %s:%dn", inet_ntoa(serv_addr.sin_addr),ntohs(serv_addr.sin_port));
write(connfd, sendBuff, strlen(sendBuff));
close(connfd);
sleep(1);
}
}