网络编程中常见地址结构与转换(IPv4/IPv6)

1. sockaddr/sockaddr_in/in_addr (IPv4)、sockaddr6_in/in6_addr/addrinfo (IPv6)

    struct sockaddr {
unsigned short sa_family;     /* address family, AF_xxx */
char sa_data[14];  /* 14 bytes of protocol address */
}; struct sockaddr_in {
short int sin_family; /* Address family AF_INET */
unsigned short int sin_port;    /* Port number */
struct in_addr sin_addr;   /* Internet address */
unsigned char sin_zero[8];    /* Same size as struct sockaddr */
};
struct in_addr {
unsigned long s_addr;   /* Internet address */
}; struct sockaddr_in6 {
sa_family_t sin6_family;   /* AF_INET6 */
in_port_t sin6_port; /* transport layer port # */
uint32_t sin6_flowinfo;   /* IPv6 traffic class & flow info */
struct in6_addr sin6_addr;     /* IPv6 address */
uint32_t sin6_scope_id;    /* set of interfaces for a scope */
};
struct in6_addr {
uint8_t s6_addr[16];   /* IPv6 address */
}; struct addrinfo{
int ai_flags; /* AI_PASSIVE,AI_CANONNAME,AI_NUMERICHOST */
int ai_family; /* AF_INET,AF_INET6 */
int ai_socktype;   /* SOCK_STREAM,SOCK_DGRAM */
int ai_protocol;   /* IPPROTO_IP, IPPROTO_IPV4, IPPROTO_IPV6 */
size_t ai_addrlen;    /* Length */
char *ai_cannoname;       /* */
struct sockaddr *ai_addr;        /* struct sockaddr */
struct addrinfo *ai_next;      /* pNext */
}

2. 与IP地址相关的常用网络编程函数

2.1 地址转化函数

IPv4中,可使用inet_ntoa/inet_aton来转化字符串形式表示的IPv4地址和数字形式表示的IPv4地址。此两函数不适用于IPv6地址转换。在Linux环境下使用inet_ntoa/inet_atoa时,需加头文件:

#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>

函数原型:

int inet_aton(const char * cp,struct in_addr *inp);
char * inet_ntoa(struct in_addr in);

举例:

#include <stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h> int main(int aargc, char* argv[])
{
struct in_addr addr1;
ulong l1;
l1 = inet_addr("192.168.0.74");
memcpy(&addr1, &l1, sizeof(l1));
printf("%s\n", inet_ntoa(addr1));
   if(inet_aton("127.0.0.1", &addr1)){
    printf("inet_aton:ip=%lu\n",ntohl(inp.s_addr));
   }
   else{
    printf("inet_aton return -1 when 255.255.255.255\n");
   }
   return 0;
}

IPv6中,使用inet_ntop/inet_pton来转化字符串形式表示的IPv6地址和数字形式表示的IPv6地址。IPv4中也可使用这两个函数。

函数原型:

int inet_pton(int af, const char *src, void *dst);         
//这个函数转换字符串到网络地址,第一个参数af是地址族,转换后存在dst中 af的值可为AF_INET (代表使用IPv4协议)或AF_INET6(代表作用IPv6协议) const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt);
//这个函数转换网络二进制结构到ASCII类型的地址,参数的作用和上面相同,只是多了一个参数socklen_t cnt,
//他是所指向缓存区dst的大小,避免溢出,如果缓存区太小无法存储地址的值,则返回一个空指针,并将errno置为ENOSPC

举例:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h> int main(int argc, char **argv)
{
unsigned char buf[sizeof(struct in6_addr)];
int domain, s;
char str[INET6_ADDRSTRLEN]; if(argc != 3){
fprintf(stderr, "usage: %s {i4|i6|<num>} string/n", argv[0]);
exit(EXIT_FAILURE);
} domain = (strcmp(argv[1], "i4") == 0) ? AF_INET:(strcmp(argv[1], "i6") == 0) ? AF_INET6 : atoi(argv[1]); //IP字符串 ——》网络字节流
s = inet_pton(domain, argv[2], buf);
if(s<=0){
if(0 == s)
fprintf(stderr, "Not in presentation format/n");
else
perror("inet_pton");
exit(EXIT_FAILURE);
} //网络字节流 ——》IP字符串
if(inet_ntop(domain, buf, str, INET6_ADDRSTRLEN) == NULL){
perror("inet ntop/n");
exit(EXIT_FAILURE);
} printf("%s/n", str);
exit(EXIT_SUCCESS);
}
上一篇:javascript – 如何为凸包算法设置中间步骤的动画?


下一篇:Java中用FileInputStream和FileOutputStream读写txt文件,文件内容乱码的问题,另附循环代码小错误