我目前正在开发一个接收客户端请求的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是否有这样的信息,我可以将它留在这个十六进制形式并返回它?
答案 0 :(得分:13)
假设是IPv4。
获取sockaddr_storage
或sockaddr
结构的地址并将其转换为IPv4版本sockaddr_in
后,您可以访问IPv4地址的各个字节。
struct sockaddr_in *sin = (struct sockaddr_in *)&their_addr;
然后你可以获取s_addr
成员的地址,该成员是一个32位值(in_addr_t
),它保存ip地址的4个字节(按网络字节顺序)并将其转换为指针到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]);
答案 1 :(得分:5)
您希望使用getnameinfo()
功能:
int getnameinfo(const struct sockaddr *sa, socklen_t salen,
char *host, size_t hostlen,
char *serv, size_t servlen, int flags);
E.g:
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);
答案 2 :(得分:0)
这是一个简单的不可变类,出于您在问题中提到的相同目的,我使用了它:
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);
}
};