您的位置:首页 > 理论基础 > 计算机网络

Linux 网络编程: gethostbyname( ), getservbyname( )

2015-10-05 14:26 801 查看

前言

最近在学习网络编程,这篇文章来自于我的同学震东,发这个文章也是为了记录一下学习

gethostbyname( )

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

参数:域名

返回值:

一个 hostent 结构体的地址(也就是一个指向 hostent 结构体的指针)

0(如果域名不能解析成 IP 地址)

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

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 。

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

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

现在那就来实战一下吧:

#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++) {
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 结构体的说明:

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"

现在就来实战一下吧:

#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;
}

编译运行:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: