####socket網絡編程接口
socket的地址是結構體sockaddr
代碼以下
struct sockaddr{
sa_family_t sa_family;
char sa_data[14];
}
sa_family 成員是地址族類型(sa_family_t)變量。
地址族類型一般與協議族類型對應
1.兩者對應表linux
協議族 | 地址表 | 描述 |
---|---|---|
PF_UNIX | AF_UNIX | UNIX本地域協議族 |
PF_INET | AF_INET | TCP/IPv4協議族 |
PF_INET6 | AF_INET6 | TCP/IPv6協議族 |
兩者定義在bits/socket.h頭文件,值同樣,因此兩者常常混用編程
sa_data成員用於存放socket地址值
可是不一樣的協議族的地址值具備不一樣的含義和長度網絡
2.協議族及其地址值 |
協議族 | 地址值含義和長度 |
---|---|---|
PF_UNIX | 文件的路徑名,長度可達108字節 |
協議族 | 地址值含義和長度 |
---|---|
PF_INET | 16bit 端口號和32bit IPv4地址 |
PF_UNIX | 文件的路徑名,長度可達108字節 |
PF_INET6 | 16bit 端口號,32bit流標識,128bitIPv6地址,32bit範圍ID,共26字節 |
問題:14字節的sa_data沒法容納多數協議族的地址值,所以linux定義了新的通用socket地址結構體
struct sockaddr_storage{socket
sa_family_t sa_family unsigned long int __sa_align; char __ss_padding[128-sizeof(__ss_align)];
}ide
ssalign:用於內存對齊函數
3.專用socket地址
問題: 上述通用結構體很很差用,設置與獲取IP地址和端口號須要執行繁瑣的位操做。
解決:linux爲各個協議族提供了專門的socket地址結構體
3.1
UNIX本地域協議族:
struct sockaddr_un{
sa_family_t sin_family; //地址族AF_UNIX
char sun_path[108]; // 文件路徑名code
} TCP/IP協議族有了兩個專用 struct socketaddr_in{ sa_family_t sin_family; //地址族AF_INET u_int16_t sin_port; //端口號,要用網絡字節序表示 struct in_addr sin_addr; //Ipv4地址結構體 } struct in_addr{ u_int32_t s_addr;//IPv4地址,要用網絡字節序表示 } struct socketaddr_in6{ sa_family_t sin6_family; //地址族AF_INET u_int16_t sin6_port; //端口號,要用網絡字節序表示 u_int32_t sin6_flowinfo; //流信息,設置爲0 struct in6_addr sin6_addr; //Ipv6地址結構體 u_int32_t sin6_scope_id; //scope_id實驗用 } struct in6_addr{ unsigned char sa_addr[16];//IPv6地址,要用網絡字節序表示 } 使用說明:全部專用地址類型的變量在實際使用中都須要轉化爲通用socket地址類型sockaddr(強制轉換) 緣由:全部的socket變成接口使用的地址參數的類型爲sockaddr 4.IP地址轉換函數 IPv4:點分十進制字符串 IPv6:16進制字符串 in_addr_t inet_addr (const char * strptr); int inet_aton(const char * cp, struct in_addr * inp); char * inet_ntoa(struct in_addr in);