inet_ntoa、 inet_aton、inet_addr

inet_addr()

  简述:将一个点间隔地址转换成一个in_addr。
  #include <winsock.h>
  unsigned long PASCAL FAR inet_addr( const struct FAR* cp);
  cp:一个以Internet标准“.”间隔的字符串。
  注释:
  本函数解释cp参数中的字符串,这个字符串用Internet的“.”间隔格式表示一个数字的Internet地址。返回值可用作Internet地址。所有Internet地址以网络字节顺序返回(字节从左到右排列)。
  Internet地址用“.”间隔的地址可有下列几种表达方式:
  a.b.c.d,a.b.c,a.b,a
  当四个部分都有定值时,每个都解释成一个字节数据,从左到右组成Internet四字节地址。请注意,当一个Internet地址在Intel机器上表示成一个32位整型数时,则上述的字节为“d.c.b.a”。这是因为Intel处理器的字节是从右向左排列的。
  请注意:只有Berkeley支持下述表达法,Internet其余各处均不支持。考虑到与软件的兼容性,应按规定进行使用。
  对一个三部分地址,最后一部分解释成16位数据并作为网络地址的最右两个字节。这样,三部分地址便很容易表示B组网络地址,如“128.net.host”.
  对一个两部分地址,最后一部分解释成24位数据并作为网络地址的最右三个字节,这样,两部分地址便很容易表示C组网络地址,如“net.host”。
  对仅有一个部分的地址,则将它的值直接存入网络地址不作任何字节的重组。
  返回值:
  若无错误发生,inet_addr()返回一个无符号长整型数,其中以适当字节顺序存放Internet地址。如果传入的字符串不是一个合法的Internet地址,如“a.b.c.d”地址中任一项超过255,那么inet_addr()返回INADDR_NONE。
  参见:

  inet_ntoa().

inet_addr()函数的实现

输入是点分的IP地址格式(如A.B.C.D)的字符串,从该字符串中提取出每一部分,转换为ULONG,假设得到4个ULONG型的A,B,C,D,
ulAddress(ULONG型)是转换后的结果,
ulAddress = D<<24 + C<<16 + B<<8 + A(网络字节序),即inet_addr(const char *)的返回结果
另外,我们也可以得到把该IP转换为主机序的结果,转换方法一样
A<<24 + B<<16 + C<<8 + D

inet_ntoa() 

简述:

  将网络地址转换成“.”点隔的字符串格式。

  #include <winsock.h>

  char FAR* PASCAL FAR inet_ntoa( struct in_addr in);

  in:一个表示Internet主机地址的结构。

  注释:

  本函数将一个用in参数所表示的Internet地址结构转换成以“.” 间隔的诸如“a.b.c.d”的字符串形式。请注意inet_ntoa()返回的字符串存放在WINDOWS套接口实现所分配的内存中。应用程序不应假设该内存是如何分配的。在同一个线程的下一个WINDOWS套接口调用前,数据将保证是有效。

  返回值:

  若无错误发生,inet_ntoa()返回一个字符指针。否则的话,返回NULL。其中的数据应在下一个WINDOWS套接口调用前复制出来。

  测试代码如下

  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,addr2;

  ulong l1,l2;

  l1= inet_addr("192.168.0.74");

  l2 = inet_addr("211.100.21.179");

  memcpy(&addr1, &l1, 4);

  memcpy(&addr2, &l2, 4);

  printf("%s : %s/n", inet_ntoa(addr1), inet_ntoa(addr2)); //注意这一句的运行结果

  printf("%s/n", inet_ntoa(addr1));

  printf("%s/n", inet_ntoa(addr2));

  return 0;

  }

  实际运行结果如下:

  192.168.0.74 : 192.168.0.74 //从这里可以看出,printf里的inet_ntoa只运行了一次。

  192.168.0.74

  211.100.21.179

  inet_ntoa返回一个char *,而这个char *的空间是在inet_ntoa里面静态分配的,所以inet_ntoa后面的调用会覆盖上一次的调用。第一句printf的结果只能说明在printf里面的可变参数的求值是从右到左的,仅此而已。

inet_aton()

inet_aton是一个改进的方法来将一个字符串IP地址转换为一个32位的网络序列IP地址。这个函数的概要如下:
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int inet_aton(const char *string, struct in_addr *addr);

inet_aton函数接受两个参数。参数描述如下:

1 输入参数string包含ASCII表示的IP地址。
2 输出参数addr是将要用新的IP地址更新的结构。

如果这个函数成功,函数的返回值非零。如果输入地址不正确则会返回零。使用这个函数并没有错误码存放在errno中,所以他的值会被忽略。

对于这个函数有一点迷惑的就是这个函数调用所需要的两个参数。如果我们定义了一个AF_INET套接口地址:

struct sockaddr_in adr_inet;    /* AF_INET */

提供给inet_aton函数调用的参数指针为 &adr_inet.sin_addr

下面这个程序使用inet_aton函数,而不是我们在前面所谈到的in_addr函数。
/*
 * inetaton.c
 *
 * Example using inet_aton
 */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

/*
 * this function reports the error and
 * exits back to the shell
 */
static void bail(const char *on_what)
{
    fputs(on_what,stderr);
    fputs("/n",stderr);
}

int main(int argc,char **argv)
{
    int z;
    struct sockaddr_in adr_inet;    /* AF_INET */
    int len_inet;            /* length */
    int sck_inet;            /* Socket */

/* Create a Socket */
    sck_inet = socket(AF_INET,SOCK_STREAM,0);

if(sck_inet == -1)
    bail("Socket()");

/* Establish address */
    memset(&adr_inet,0,sizeof adr_inet);
    adr_inet.sin_family = AF_INET;
    adr_inet.sin_port = htons(9000);

if( !inet_aton("127.0.0.1",&adr_inet.sin_addr))
    bail("bad address");

len_inet = sizeof adr_inet;

/* Bind it to the socket */
    z = bind(sck_inet,(struct sockaddr *)&adr_inet,len_inet);

if(z == -1)
    bail("bind()");

/* Display our socket address */
    system("netstat -pa --tcp 2>/dev/null"
        " | grep inetaton");

return 0;
}
程序的运行结果如下:
S$ ./inetaton
tcp 0 0 127.0.0.23:9000 *:* CLOSE 1007/inetaton

时间: 2024-08-06 20:07:38

inet_ntoa、 inet_aton、inet_addr的相关文章

【网络编程】inet_addr、inet_ntoa、inet_aton、inet_ntop和inet_pton区分

先上一张图 1.把ip地址转化为用于网络传输的二进制数值 int inet_aton(const char *cp, struct in_addr *inp); inet_aton() 转换网络主机地址ip(如192.168.1.10)为二进制数值,并存储在struct in_addr结构中,即第二个参数*inp,函数返回非0表示cp主机有地有效,返回0表示主机地址无效.(这个转换完后不能用于网络传输,还需要调用htons或htonl函数才能将主机字节顺序转化为网络字节顺序) in_addr_t

【网络】IP地址格式转换(htonl、ntohl;inet_addr、inet_ntoa)

1.htonl ()和ntohl( ) u_long PASCAL FAR ntohl (u_long netlong); u_short PASCAL FAR ntohs (u_short netshort); ntohl( )-----网络顺序转换成主机顺序 u_long PASCAL FAR htonl (u_long hostlong); u_short PASCAL FAR htons (u_short hostshort); htonl ()-----主机顺序转换成网络顺序 2.in

Unix网络编程随手记——IP处理函数inet_aton()、gethostbyname()等

IP地址实质上就是一个32位的无符号整数,用如下结构体存放 1 struct in_addr 2 { 3 unsigned int s_addr; 4 }; 由于历史原因,虽然IP地址只是个标量,却用一个结构体来存储. 由于主机可以有不同的主机字节顺序,即大端机或小端机.但TCP/IP定义了统一的网络字节顺序,大端字节顺序. Unix提供了两个函数可在主机字节和网络字节间实现转换: 一.htonl()和ntohl() 在Linux系统下: #include <arpa/inet.h> 有些系统

socket编程以及select、epoll、poll示例详解

socket编程socket这个词可以表示很多概念,在TCP/IP协议中“IP地址 + TCP或UDP端口号”唯一标识网络通讯中的一个进程,“IP + 端口号”就称为socket.在TCP协议中,建立连接的两个进程各自有一个socket来标识,那么两个socket组成的socket pair就唯一标识一个连接. 预备知识 网络字节序:内存中多字节数据相对于内存地址有大端小端之分,磁盘文件中的多字节数据相对于文件中的偏移地址也有大端小端之分.网络数据流同样有大端小端之分,所以发送主机通常将发送缓冲

MySQL数据库(7)_用户操作与权限管理、视图、存储过程、触发器、基本函数

用户操作与权限管理 MySQL用户操作 创建用户 方法一: CREATE USER语句创建 CREATE USER "用户名"@"IP地址" IDENTIFIED BY "密码"; 方法二: INSERT语句创建 INSERT INTO mysql.user(user,host, password,ssl_cipher,x509_issuer,x509_subject) VALUES('用户名','IP地址',password('密码'),'',

Linux多线程编程----IO【select、poll、epoll】

IO操作多   速度就下降 IO数据的 读和写 IO的完成 必须等到 读事件(如磁盘 拷贝  每次要从磁盘查找数据) 和 写事件 (允许写 如写太快 写满就要马上阻塞)的就绪 IO是否高效 :主要看一次IO中 等的时间的比例的多少 (等的时间比例越少  越高效) 就像钓鱼分两步:1 等 2 钓   (评价钓鱼技术高效 是 等的时间少 钓的次数多) 5中IO模型 1 阻塞式IO: 等的时候自己等   , 数据搬迁也是由自己来操作, IO事件就绪时 自己处理.期间什么也不做 2 非阻塞IO :(轮询

UNIX网络编程:I/O复用技术(select、poll、epoll)

Unix下可用的I/O模型一共有五种:阻塞I/O .非阻塞I/O .I/O复用 .信号驱动I/O .异步I/O.此处我们主要介绍第三种I/O符复用. I/O复用的功能:如果一个或多个I/O条件满足(输入已准备好读,或者描述字可以承接更多输出)时,我们就被通知到.这就是有select.poll.epoll实现. I/O复用应用场合: 1.当客户处理多个描述字时(一般是交互式输入和网络套接口),必须使用I/O复用.在这前一段中已做描述. 2.一个客户同时处理多个套接口是可能的,但很少出现. 3.如果

网络编程1 网络开发基础(Socket、TCP、UDP)

OSI七层网络模型 1.网络的七层结构:物理层.数据链路层.网络层.传输层.会话层.表示层和应用层. 2.在网络通信的发送端,其通信数据每到一个通信层,都会被该层协议在数据中添加一个包头数据.而在接收方恰恰相反,数据通过每一层时都会被该层协议剥去相应的包头数据. TCP/IP 1.IP地址 IP地址是指互联网协议地址(Internet Protocol Address,又译为网际协议地址).IP地址时IP协议提供的一种统一的地址格式,它为互联网上的每一个网络和每一个主机分配一个逻辑地址,以此来屏

(十一)socket、connect、bind函数详解

一.socket函数 1.头文件: #include <sys/types.h> /* See NOTES */ #include <sys/socket.h> 2.函数原型: int socket(int domain, int type, int protocol); socket函数类似于open,用来打开一个网络连接,如果成功则返回一个网络文件描述符(int类型),之后我们操作这个网络连接都通过这个网络文件描述符. dimain:域,网络域,网络地址范围(IPV4或IPV6