Linux Socket学习--地址转换函数

 一个IP地址是由小数点分开的十进制数表示的,我们称之为点分十进制表示法。其中每一个十进制数代表一个字节的无符号数值(按照网络字节序)因为每个字节都是无符号的8位数值,这就限制了每一个字节所能表示的范围是0~255.

Internet地址分类

     一个Internet地址是由网络地址和主机地址构成的。

     我们知道IP地址由32位二进制构成,但是网络地址和主机地址之间的界限并不是固定的,而界限的确定取决于地址的分类,下表总结了IP地址分类的方法:

理解网络掩码:

     网络掩码的作用在于把网络地址从IP地址从提取出来,实际上代表网络掩码的IP号和某一特定的IP地址进行“按位与”操作。

     如果我们需要建立自己的IP网络,那么就必须确定网络掩码,如下图:

软件有时候需要提供将IP地址进行分类的能力,下面的例子就展示了这个操作:

---------UNDONE(此处代码待添加)

分配IP地址

     ip地址是通过InterNIC分配给不同的组织和个人的。但是并非所有的IP地址都是可以用来分配的,其中也有一些私有的,或者说是预留出来的。

私有IP

      一般IP地址都需要在InterNIC进行登记,但是如果你的系统没有直接连接到Internet,那么久不需要全球唯一的地址,这个时候就可以使用私有IP地址来替代。

      RFC1597就是描述怎么分配私有IP地址的文档。

      

最后到底选择哪一种IP地址在很大的程度上取决于所需要建立的网络数量,分离的网段以及每个网段上主机的数目。

保留IP地址

   保留IP地址的数量很多,在RFC1166文档中有记录。下面就是一个保留IP地址的实例:

操作IP地址

    inet_addr()函数的语法如下:

#include <sys/socket.h>
#include <neiinet/in.h>

#include <arpa/inet.h>

unsigned long inet_addr(const char * string);

        这个函数使用string作为输入参数,并将这个点分十进制的IP地址转换为32位的二进制表示法,函数的返回值就是这个32位的二进制的网络字节序。当然如果string不是一个有效的点分十进制IP地址,函数返回INADDR_NONE。另外需要注意的是,当inet_addr函数返回INADDR_NONE的时候,它并没有建立一个有效的errno值,所以当函数返回错误的时候,不要去测试errno的值。

        下面的这个例子展示了如何使用函数inet_addr。

       。。。。。。。。。UNDONE(此处代码待添加)

       注意:

       在新程序中避免使用inet_addr函数,而应该使用inet_aton函数作为代替。因为对于inet_addr函数来说,即使输入的参数是有效的IP地址:255.255.255.255,他的返回值仍然是INADDR_NONE。

下面我们来说说inet_aton函数:

     inet_aton函数是将字符串形式的IP地址转换为网络字节序的32位IP地址的改进形式。语法如下:

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

int inet_aton(const char* string, struct in_addr*addr);

参数string表示点分十进制IP地址的ASCII表示。输出参数addr是一个被新的IP地址跟新的结构。函数调用成功返回非0值。失败返回0.当然他也没有建立一个有效的errno值。下面的代码展示了inet_aton函数的用法:

。。。。。。。。。UNDONE(此处代码待添加)

下面我们来看看inet_ntoa函数

        有时候当用户连接到你的服务器的时候,需要知道他的IP地址,系统提供了inet_ntoa函数将32位的二进制IP地址表示转换为点分十进制的字符串形式:

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

char* inet_ntoa(struct in_addr addr);

下面的代码展示了如何使用inet_ntoa函数:

。。。。。。。。。UNDONE(此处代码待添加)

       需要注意的是inet_ntoa函数的返回值直到下次调用前一直有效。所以如果在线程中使用inet_ntoa的时候,一定要确保每次只有一个线程调用本函数。否则一个线程的返回的结构可能被其他线程返回的结果所覆盖。

接下来我们来看看inet_network函数。

        当我们需要用网络掩码将IP地址中的网络位或者主机位提取出来的时候,如果能将点分十进制的IP地址转换为主机字节序的32位二进制IP地址形式就方便了,而inet_network函数的作用就是如此、语法如下:

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

unsigned long inet_network(const char* addr);

函数的输入参数是存储在字符串addr中的点分十进制IP地址,返回值是主机字节序的32位二进制地址,但是如果输入参数不正确,返回值是0xFFFFFFFF.

    以主机字节序的形式返回结果可以保证用户安全的使用网络掩码,因为如果返回值是网络字节序的话,那么不同的cPu平台所使用的网络掩码和程序代码就会有差异。下面的例子展示了如何使用inet_network函数。

。。。。。。。。。UNDONE(此处代码待添加)

我们来看看inet_lnaof函数。

        函数inet_lnaof函数是将套接口地址中的IP地址(网络字节序)转换为没有网络位的主机ID(主机字节序),这个函数为我们省去了很多的麻烦,因为我们不需要对IP地址进行分类,再将主机为从IP地址中提取出来。函数语法如下:

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

unsigned long inet_lnaof(struct in_addr addr);

下面的表提供了一些可以作为inet_lnaof函数输入的典型例子和返回值:

我们再来看看inet_netof函数

      inet_lnaof函数返回的是主机ID,而inet_netof函数返回的是网络ID,函数语法如下:

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

unsigned long inet_netof(struct in_addr addr);

下表展示了一些例子:

然后我们再来看看inet_makeaddr函数吧。根据之前的内容我们知道使用inet_netof函数和inet_lnaof函数,我们就可以把IP地址中的主机为和网络位分别提取出来,有时候我们还需要根据提取出来的主机位和网络为合并为一个新的IP地址。这个时候我们就可以使用inet_makeaddr函数。

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

struct in_addr inet_makeaddr(int net , int host);

下面的例子展示了上面讲到的三个函数:

。。。。。。。。。UNDONE(此处代码待添加)

时间: 2024-09-25 13:49:29

Linux Socket学习--地址转换函数的相关文章

1.socket编程:socket编程,网络字节序,函数介绍,IP地址转换函数,sockaddr数据结构,网络套接字函数,socket相关函数,TCP server和client

 1  Socket编程 socket这个词可以表示很多概念: 在TCP/IP协议中,"IP地址+TCP或UDP端口号"唯一标识网络通讯中的一个进程,"IP 地址+端口号"就称为socket. 在TCP协议中,建立连接的两个进程各自有一个socket来标识,那么这两个socket组成的socket pair就唯一标识一个连接.socket本身有"插座"的意思,因此用来描述网络连 接的一对一关系. TCP/IP协议最早在BSD UNIX上实现,

linux网络编程之socket(一) socket概述和字节序、地址转换函数

一.什么是socket socket可以看成是用户进程与内核网络协议栈的编程接口. socket不仅可以用于本机的进 程间通信,还可以用于网络上不同主机的进程间通信. socket API是一层抽象的网络编程接口,适用于各种底层网络协议,如IPv4.IPv6,以及以后要讲的UNIX Domain Socket.然而,各种网络协议的地址格式并不相同,如下图所示: IPv4和IPv6的地址格式定义在netinet/in.h 中,IPv4地址用sockaddr_in结构体表示,包括16位端口号和32位

Linux Socket学习--域和地址族

先来说说无名套接口吧:        套接口不一定需要地址,比如函数socketpair就生成了一对相互连接但是没有地址的套接口,这就是所谓的无名套接口.        有时候也会有这样的情况,在相互连接的两个太接口中有一个套接口不需要地址,例如当连接到一个远程的套接口的时候,虽然必须确定远程套接口的地址,但是发出调用的本地套接口却可以是匿名的.         有时候虽然需要一个地址进行通信,但是并不关心这个地址具体是什么,这个本地地址仅仅在通信过程中保持有效.如果给他分配一个固定的地址,浪费

Linux Socket学习--面向非连接的协议

套接口的通信有2中基本方式:面向连接和面向非连接的通信. 面向非连接指的是通信之前不需要建立连接,我们生成了一个非连接的套接口,就可以向任何愿意接受我们消息的套接口发送消息,而且每一个消息都可以被重定向到不同的套接口. 我们之前的文章中提供socketpair函数,但是我们当时没有说这个函数是使用面向连接协议来生成套接口对的. sendto()函数介绍     sendto函数容许我们写一个数据报,并且同时制定接受者的地址.语法如下: #include <sys/types.h> #inclu

地址转换函数

IPv4 #include <arpa/inet.h>   点分十进制数串(例如192.168.1.1) ===(转换)===> 32位网络字节序地址 int inet_aton(const char* strptr, struct in_addr *addrptr); 返回: 1----串有效 0----串有错   in_addr_t inet_addr(const char* strptr); 返回: 成功,32位二进制的网络字节序地址 失败,返回INADDR_NONE   32位网

Linux Socket学习--为套接口绑定地址

  当我们使用socket函数创建一个套接口之后,这个套接口就处于无名状态,虽然之前我们提到即使没有地址,套接口也能使用,但是这个只限于套接口对在同一个Linux内核中,如果位于两台不同的主机的套接口需要连接,而又没有地址,那么就无法工作.    函数bind的作用就是为无名套接口分配地址,语法如下: #include <sys/types.h> #include <sys/socket.h> int bind(int sockfd, struct sockaddr* my_add

Linux Socket学习--套接口的类型和协议

  我们首先来说一下PF_INET和AF_INET,虽然标准提倡在指定demain参数的时候,优先使用PF_INET,但是大量已经编写的c代码遵循旧的协议.目前情况是AF_UNIX=PF_UNIX,AF_INET=PF_INET.但是将来是不是这样不好说啊.           不同于socketpari函数的demain参数只能指定为PF_LOCAL,socket函数可以用于生成所支持的任何协议族的套接口,语法如下: #include <sys/types.h> #include <s

Linux Socket学习--域和套接口简介

       套接口创建后,就如同一个文件描述符,我们可以使用同样的IO函数进行读写,关闭操作.其实,和引用一个已经打开的文件一样,套接口也是通过文件描述符来引用的,而且两者的文件描述符共享一个"数字空间",比如说不能既打开一个文件描述符为4的套接口,又打开一个文件描述符为4的文件.        套接口和已经打开的文件的区别:        1.不能在套接口上调用函数lseek()当然对于管道也不能调用这个函数.        2.套接口可以和网络地址关联,但是文件和管道却不可以.

【linux网络编程】网络字节序、地址转换

网络字节序 故事的起源 "endian"这个词出自<格列佛游记>.小人国的内战就源于吃鸡蛋时是究竟从大头(Big-Endian)敲开还是从小头(Little-Endian)敲开,由此曾发生过六次叛乱,其中一个皇帝送了命,另一个丢了王位. 我们一般将"endian"翻译成"字节序",将 Big-Endian 和 Little-Endian 称作"大端格式"和"小端格式". 字节序 字节序是指多字节