Linux gethostbyname() 和 getservbyname() 函数的使用

前言

最近在学习网络编程,用到几个应该比较常用的网络编程函数,所以写篇博客来记录一下,毕竟学得快忘得也快。国庆节在宿舍写着博客看着各个景点人山人海倒也快哉~

gethostbyname( )

这个函数可以返回给定域名的域名信息。

参数:域名

返回值:

  • 一个 hostent 结构体的地址(也就是一个指向 hostent 结构体的指针)
  • 0(如果域名不能解析成 IP 地址)

netdb.h 头文件我们可以找到 hostent 结构体的说明:

1
2
3
4
5
6
7
8
struct hostent {
char *h_name; /*official host name */
char **h_aliases; /*other aliases */
int h_addrtype; /*address type */
int h_length; /* address length */
char **h_addr_list; /* list of addresses */
};
#define h_addr h_addr_list[0]

上面结构体中的 **h_addr_list 是一个二进制整数的链表,输出的时候要用 inet_ntop() 函数转换成点分十进制。

inet_ntop()

inet_ntop() 和 inet_pton() 都是IP地址转换函数,可以在将IP地址在“二进制整数”和“点分十进制”之间转换。而且,这2个函数能够处理 ipv4 和 ipv6 。

1
const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt);

这个函数转换网络二进制结构到ASCII类型的地址,参数的作用和inet_pton相同,只是多了一个参数 socklen_t cnt ,他是所指向缓存区 dst 的大小,避免溢出,如果缓存区太小无法存储地址的值,则返回一个空指针,并将 errno 置为 ENOSPC 。

现在那就来实战一下吧:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include <stdio.h>
#include <netdb.h>
#include <arpa/inet.h>

int main(int argc, char **argv)
{
struct hostent *hptr;
char *name, **pptr, str[32];
int count = 0;

if (argc < 2) {
printf("The arguments is not enough!");
return -1;
}

name = argv[1];
hptr = gethostbyname(name);
if (hptr) {
printf("the offical name is %s.\n", hptr->h_name);
for(pptr = hptr->h_aliases; *pptr != NULL; pptr++) {
printf("the alias name is %s\n", *pptr);
}

switch (hptr->h_addrtype) {
case AF_INET:
printf("the address type is AF_INET.\n");
break;
case AF_INET6:
printf("the address type is AF_INET6.\n");
break;
default:
break;
}

printf("the address length is %d Bytes.\n", hptr->h_length);

for (pptr = hptr->h_addr_list; *pptr != NULL; pptr++) { //**pptr后移四个字节,即地址后移四位
count ++;
printf("the %dth address is %s.\n", count, inet_ntop(hptr->h_addrtype, *pptr, str, sizeof(str))); //即将转换成的点分十进制存到字符串 str 中返回,溢出则返回空指针
}
} else {
printf("Error!\n");
}

return 0;
}

编译运行:

getservbyname

这个函数可以返回给定服务名和协议名的相关服务信息。

参数:服务名和协议名

返回值:

  • 一个指向 servent 结构体的指针
  • 空指针(发生错误)

netdb.h 头文件我们可以找到 servent 结构体的说明:

1
2
3
4
5
6
struct servent {
char *s_name; /*official service name */
char **s_aliases; /*other aliases */
int s_port; /*port for this service */
char **s_proto; /* protocol to use */
};

返回的结构体中的端口号是按网络字节顺序保存的整数,输出的时候要用 ntohs() 函数转换按主机顺序保存的整数。

ntohs()

网络字节顺序NBO(Network Byte Order):按从高到低的顺序存储,在网络上使用统一的网络字节顺序,可以避免兼容性问题。

主机字节顺序(HBO,Host Byte Order):不同的机器HBO不相同,与CPU设计有关,数据的顺序是由cpu决定的,而与操作系统无关。

网络字节顺序与本地字节顺序之间的转换函数:

htonl()–”Host to Network Long”
ntohl()–”Network to Host Long”
htons()–”Host to Network Short”
ntohs()–”Network to Host Short”

现在就来实战一下吧:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#include <stdio.h>
#include <netdb.h>

int main (int argc, char* argv[])
{
struct servent *sptr;
char *service, *protocol;

if (argc < 3) {
printf("The arguments is not enough!\n");
return -1;
}

service = argv[1];
protocol = argv[2];
sptr = getservbyname(service, protocol);
if (sptr) {
printf("the port of service %s using %s protocol is %d.\n", sptr->s_name, protocol, ntohs(sptr->s_port)); //将网络字节顺序的端口值转换成主机顺序
} else {
printf("Error!\n");
}

return 0;
}

编译运行:

疏影横斜水清浅,暗香浮动月黄昏