7

我目前正在使用接收来自客户端的请求的 UDP 服务器。我收到的数据报是一个 5 个元素长的字节 (char) 数组,最后两个元素是端口号。

最终,该服务器将不得不在自己的数据报中返回 IP 地址和端口号。

我已经知道如何使用 inet_ntop 和我已连接并从中接收到的 sockaddr 结构来打印出 ip,但它返回的字符串不是我想要的格式。例如:

string1 = inet_ntop(their_addr.ss_family,get_in_addr(
    (struct sockaddr *)&their_addr),s, sizeof s);

返回:

127.0.0.1

或者:

[1][2][7][.][0][.][0][.][1]

当我需要类似的东西时:

[127][0][0][1]

我应该使用某种字符和数组操作来制作我的 4 元素字节数组吗?或者 sockaddr 是否以某种方式拥有这些信息,我可以将其保留为这种十六进制形式并返回它?

4

3 回答 3

13

假设为 IPv4。

在获取您的sockaddr_storagesockaddr结构的地址并将其转换为 IPv4 版本sockaddr_in之后,您可以访问 IPv4 地址的各个字节。

struct sockaddr_in *sin = (struct sockaddr_in *)&their_addr;

然后你可以获取s_addr成员的地址,它是一个 32 位值 ( in_addr_t),它包含 ip 地址的 4 个字节(按网络字节顺序),并将其转换为指向 an 的指针unsigned char,然后你可以访问该值的各个字节.

unsigned char *ip = (unsigned char *)&sin->sin_addr.s_addr;

printf("%d %d %d %d\n", ip[0], ip[1], ip[2], ip[3]);
于 2012-10-10T04:25:20.830 回答
5

您可能想使用getnameinfo()功能:

int getnameinfo(const struct sockaddr *sa, socklen_t salen,
                   char *host, size_t hostlen,
                   char *serv, size_t servlen, int flags);

例如:

struct sockaddr_storage client_addr;
socklen_t client_len = sizeof(struct sockaddr_storage);

/* Accept client request */
int client_socket = accept(server_socket, 
    (struct sockaddr *)&client_addr, &client_len);

char hoststr[NI_MAXHOST];
char portstr[NI_MAXSERV];

int rc = getnameinfo((struct sockaddr *)&client_addr, 
    client_len, hoststr, sizeof(hoststr), portstr, sizeof(portstr), 
    NI_NUMERICHOST | NI_NUMERICSERV);

if (rc == 0) 
    printf("New connection from %s %s", hoststr, portstr);
于 2013-12-05T10:47:55.547 回答
1

这是一个简单的不可变类,我用于您在问题中提到的相同目的:

class address_t {

private:
    uint16_t m_Port = 0;
    std::string m_Ip = "";

public:
    address_t(const sockaddr_in & address) {
        m_Ip = inet_ntoa(address.sin_addr);
        m_Port = ntohs(address.sin_port);
    }

    uint16_t GetPort() const { return m_Port; }
    std::string GetIp() const { return m_Ip; }
    std::string ToString() const {
        return "IP: " + m_Ip + ", Port: " + std::to_string(m_Port);
    }
};
于 2019-04-29T06:56:21.417 回答