struct hostent *gethostbyname(const char *name);
返回:若成功则为非空字符串,若出错则为NULL且设置h_errno
注意:此函数的局限只能返回IPV4地址,所以建议使用getaddrinfo()函数
本函数返回的非空指针结构体如下:
struct hostent
{
char *h_name;
char **h_aliases;
int h_addrtype;
int h_length;
char **h_addr_list;
};
hostent->h_name
表示的是主机的规范名。例如www.google.com的规范名其实是www.l.google.com。
hostent->h_aliases
表示的是主机的别名.www.google.com就是google他自己的别名。有的时候,有的主机可能有好几个别名,这些,其实都是为了易于用户记忆而为自己的网站多取的名字。
hostent->h_addrtype
表示的是主机ip地址的类型,AF_INET
hostent->h_length
表示的是主机ip地址的长度
hostent->h_addr_lisst
表示的是主机的ip地址,注意,这个是以网络字节序存储的。千万不要直接用printf带%s参数来打这个东西,会有问题的哇。所以到真正需要打印出这个IP的话,需要调用inet_ntop()。
const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt) :
这个函数,是将类型为af的网络地址结构src,转换成主机序的字符串形式,存放在长度为cnt的字符串中。返回指向dst的一个指针。如果函数调用错误,返回值是NULL。
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define SERVPORT 80
int main(int argc, char **argv)
{
int i, sockfd;
struct hostent *host;
struct sockaddr_in serv_addr;
if ((host = gethostbyname(argv[1])) == NULL) {
printf("gethostbyname error\n");
exit(0);
}
printf("official name: %s\n\n", host->h_name);
printf("address length: %d bytes\n\n", host->h_length);
printf("host name alias: \n");
for (i = 0; host->h_aliases[i]; i++) {
printf("%s\n", host->h_aliases[i]);
}
printf("\naddress list: \n");
for (i = 0; host->h_addr_list[i]; i++) {
if ((sockfd = socket(AF_INET, SOCK_STREAM,0)) == -1) {
printf("socket error\n");
exit(0);
}
// 先清零,然后用struct sockaddr_in来填值
bzero(&serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(SERVPORT);
/* h_addr_list[i]指向in_addr类型 */
serv_addr.sin_addr = *((struct in_addr *)host->h_addr_list[i]);
const char *ip = inet_ntoa(serv_addr.sin_addr);
printf("connect to %s ", ip);
// 系统调用的时候,把sockaddr_in转换成sockaddr
if (connect(sockfd, (struct sockaddr *)&serv_addr, \
sizeof(struct sockaddr)) == -1) {
printf("error\n");
exit(0);
}
printf("success\n");
}
return 0;
}