扣扣技术交流群:460189483
基于Linux C++ 开发udp通讯的一些总结,主要是记性不好,写下来防止后面又忘记了,需要重新找资料浪费时间,先了解一下基本数据结构
一、sockaddr和sockaddr_in结构
程序员不应操作sockaddr结构,sockaddr是给操作系统用的
程序员应使用sockaddr_in来表示地址,sockaddr_in区分了地址和端口,使用更方便。
struct sockaddr {
unsigned short sa_family; // 2 bytes address family, AF_xxx unsiged short
char sa_data[14]; // 14 bytes of protocol address
};
struct sockaddr_in {
short sin_family; // 2 bytes e.g. AF_INET, AF_INET6
unsigned short sin_port; // 2 bytes e.g. htons(3490)
struct in_addr sin_addr; // 4 bytes see struct in_addr, below
char sin_zero[8]; // 8 bytes zero this if you want to
};
struct in_addr {
unsigned long s_addr; // 4 bytes load with inet_pton()
};
它们结构之间表示如下图。
二、从结构sockaddr获取IP和端口。
使用结构inet_ntop获取IP地址,而不是用inet_ntoa。可能是因为64位机子引起的问题。
详细的原因参考:https://blog.csdn.net/zmxiangde_88/article/details/8157256
struct sockaddr from;
/*
...
working
...
*/
struct sockaddr_in *sock = ( struct sockaddr_in*)&from;
int port = ntohs(sock->sin_port);
#ifdef __MINGW32__ //windows上打印方式
printf("ip:port %s : %d",inet_ntoa(sock->sin_addr),port);
#else //linux上打印方式
struct in_addr in = sock->sin_addr;
char str[INET_ADDRSTRLEN]; //INET_ADDRSTRLEN这个宏系统默认定义 16
//成功的话此时IP地址保存在str字符串中。
inet_ntop(AF_INET,&in, str, sizeof(str));
printf("ip:port %s : %d",str,port);
#endif
inet_ntop接口文档
/* Convert a Internet address in binary network format for interface
type AF in buffer starting at CP to presentation form and place
result in buffer of length LEN astarting at BUF. */
extern const char *inet_ntop (int __af, const void *__restrict __cp,
char *__restrict __buf, socklen_t __len)
将其转化的结果放置到buf中。即最终转化的IP值存放在buf中。
网络字节顺序与本地字节顺序之间的转换函数:
htonl()--"Host to Network Long"
ntohl()--"Network to Host Long"
htons()--"Host to Network Short"
ntohs()--"Network to Host Short"
之所以需要这些函数是因为计算机数据表示存在两种字节顺序:NBO与HBO
网络字节顺序NBO(Network Byte Order):
按从高到低的顺序存储,在网络上使用统一的网络字节顺序,可以避免兼容性问题。
主机字节顺序(HBO,Host Byte Order):
不同的机器HBO不相同,与CPU设计有关,数据的顺序是由cpu决定的,而与操作系统无关。
如 Intelx86结构下,short型数0x1234表示为34 12, int型数0x12345678表示为78 56 34 12如IBM power PC结构下,short型数0x1234表示为12 34, int型数0x12345678表示为12 34 56 78
由于这个原因不同体系结构的机器之间无法通信,所以要转换成一种约定的数序,也就是网络字节顺序,其实就是如同powerpc那样的顺序 。在PC开发中有ntohl和htonl函数可以用来进行网络字节和主机字节的转换。
参考文章 https://blog.csdn.net/myyllove/article/details/83380209
inet_addr()、inet_pton() 将字符串形式的IP地址 -> 网络字节顺序的整型值
inet_pton()源码
int inet_pton(int family, const char *strptr, void *addrptr)
{
if (family == AF_INET) {
struct in_addr in_val;
if (inet_aton(strptr, &in_val)) {
memcpy(addrptr, &in_val, sizeof(in_val));
return (1);
}
}
errno = EAFNOSUPPOPT;
return (-1);
}
inet_ntop()源码
const char *inet_ntop(int family, const void *addrptr, char *strptr, size_t len)
{
const u_char *p = (const u_char*)addrptr;
if (family == AF_INET) {
char temp[INET_ADDRSTRLEN];
snprintf(temp, sizeof(temp), "%d.%d.%d.%d", p[0], p[1], p[2], p[3]);
if (strlen(temp) >= len) {
errno = ENOSPC;
rturn (NULL);
}
strcpy(strptr, temp);
return (strptr);
}
errno = EAFNOSUPPOPT;
return (NULL);
}
inet_addr()
转换网络主机地