gpt4 book ai didi

c - 如何在特定 IP 上打开套接字

转载 作者:太空宇宙 更新时间:2023-11-04 01:17:26 24 4
gpt4 key购买 nike

我正在尝试通过 C 创建一个简单的服务器程序,但我不想让它使用我当前的 IP,而是想提供一个特定的 IP。但事情并不奏效。

作为提供特定 IP 的第一步,我从代码中删除了以下语句:[注意]:这里,'hints' 是 struct addrinfo

的一个对象
hints.ai_flags = AI_PASSIVE;

那是因为它会自动填充我们的IP,不允许我们提供具体的IP。

然后我尝试使用 inet_pton()sin_addr(sockaddr_in 结构的成员)提供一个特定的 IP,使用语句:

inet_pton(AF_INET , host_to_connect  , &(((struct sockaddr_in *)hints.ai_addr)->sin_addr));

这里,'host_to_connect' 是包含点分十进制格式的所需 IP 地址的字符串(例如 char *host_to_connect = "192.168.22.23")

由于sockaddr结构中没有名为sin_addr的成员,所以我将其类型转换为sockaddr_in结构,然后访问sin_addr.

这是我正在处理的实际代码:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<netinet/in.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<netdb.h>
#include<arpa/inet.h>

int main(int argc , char **argv)
{
char *host_to_connect = argv[1];
char *port = argv[2];
struct addrinfo hints , *server_info;

memset(&hints , 0 , sizeof(hints));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;

inet_pton(AF_INET , host_to_connect , &(((struct sockaddr_in *)hints.ai_addr)->sin_addr));

//hints.ai_flags = AI_PASSIVE;

int status;
if((status = getaddrinfo(NULL , port , &hints, &server_info)) != 0)
{
fprintf(stderr , "Error in getaddrinfo(): %s\n" , gai_strerror(status));
return 2;
}

//creating a socket
int socketfd;
if((socketfd = socket(server_info->ai_family , server_info->ai_socktype , server_info->ai_protocol)) == -1)
{
fprintf(stderr , "Error in socket() : %s\n " , gai_strerror(socketfd));
return 2;
}

//binding port to socket
int bind_status;
if((bind_status = bind(socketfd , server_info->ai_addr , server_info->ai_addrlen)) == -1)
{
fprintf(stderr , "Error in bind() : %s\n " , gai_strerror(bind_status));
return 2;
}

//listining on the socket
int listen_status;
if((listen_status = listen(socketfd , 5)) == -1 )
{
fprintf(stderr , "Error in listen() : %s\n " , gai_strerror(listen_status));
return 2;
}

//accepting connections
int client_fd;
struct sockaddr_storage client_info;
socklen_t address_length = sizeof(client_info);
if((client_fd = accept(socketfd , (struct sockaddr *)&client_info , &address_length)) == -1)
{
fprintf(stderr , "Error in accept() : %s\n " , gai_strerror(client_fd));
return 2;
}

//sending message to client
char *message = "\n\nYou are successfully connected to the server!\n\n";
int bytes_sent;
bytes_sent = send(client_fd , message , strlen(message) , 0);
printf("\nBytes sent: %d\n" , bytes_sent);

close(socketfd);
return 0;
}

我期待它能很好地工作,但是当我运行它时,它提供了必要的参数,例如:

gcc server.c -o ./server
./server "192.168.22.23" "8989"

我得到 Segmentation Fault 作为输出。

当我尝试使用 GNU 调试器对其进行调试时,结果发现代码中使用的 inet_pton() 有问题,但我无法弄清楚。

如果有人能解决问题,我会很高兴!

最佳答案

你崩溃的原因是因为你试图取消引用 NULL 指针。

hints.ai_addr在您的 memset 之后具有 NULL 值,但您尝试使用 drerefrence表达式 &(((struct sockaddr_in *)hints.ai_addr)->sin_addr))当你调用inet_pton .这会调用 undefined behavior在这种情况下表现为崩溃。

即使你解决了这个问题,你也不应该将值放入 hints.ai_addr 中.将非 NULL 值传递给 hints 时, 该结构只能包含 ai_family 的值, ai_socktype , 或 ai_protocol领域。其余必须为 0 或 NULL。

对于您正在做的事情,您最好跳过使用 getaddrinfo并填充 struct sockaddr_in直接传递给bind .另外,gai_strerror仅用于获取 getaddrinfo 的错误字符串.对于其他套接字功能,请使用 strerrorperror :

struct sockaddr_in sin;
memset(&sin, 0, sizeof(sin));

sin.sin_family = AF_INET;
if (!inet_pton(AF_INET, host_to_connect, &sin_sin_addr)) {
fprintf(stderr, "invalid address\n");
return 2;
}
sin.sin_port = htons((uint16_t)atoi(argv[2]));

//creating a socket
int socketfd;
if((socketfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Error in socket()");
return 2;
}

//binding port to socket
int bind_status;
if((bind_status = bind(socketfd, (struct sockaddr *)&sin , sizeof(sin))) == -1)
{
perror("Error in bind()");
return 2;
}

此外,您只能绑定(bind)到系统上存在的 IP 地址。如果您有多个网络接口(interface),则可以使用它来仅监听其中一个,而不是绑定(bind)到 0.0.0.0 时监听所有接口(interface)。

关于c - 如何在特定 IP 上打开套接字,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54486239/

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