Linux网络编程--字符串IP和二进制IP转换

一:不安全的(不可重入性)inet_xxx()函数族

由于计算机理解的IP都是以二进制形式保存的,那么在网络程序设计中经常会需要字符串IP和二进制IP的转换,linux系统有一组要用于网络地址转换的函数,如下:

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

       int inet_aton(const char *cp, struct in_addr *inp);//将点分4段式的IP地址转换为结构in_addr值
       in_addr_t inet_addr(const char *cp);//将字符串转换为结构in_addr值
       in_addr_t inet_network(const char *cp);//将字符串地址的网络部分转换为结构in_addr值
       char *inet_ntoa(struct in_addr in);//将结构in_addr转为字符串
       struct in_addr inet_makeaddr(int net, int host);//将网络地址和主机地址合成为IP地址,返回值是in_addr值
       in_addr_t inet_lnaof(struct in_addr in);//获得地址的主机部分
       in_addr_t inet_netof(struct in_addr in);//获得地址的网络部分

以上部分函数存在缺陷,例如:

inet_ntoa函数返回值是一个指向字符串的指针,此内存会在每次调用inet_nota函数的时候被覆盖掉,如果不及时拿走数据就会出现不可预料的错误,因此函数是不安全的,存在某种隐患;

inet_addr,inet_network函数的返回值是-1时表示错误,占用了地址255.255.255.255的值,存在缺陷,埋下了隐患。

下面用一段代码介绍使用方法和隐患的部分:

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>

int main(int argc, char *argv[])
{
	struct in_addr ip,local,network;
	char addr1[]="192.168.1.1";		/* 网络地址字符串 */
	char addr2[]="255.255.255.255";
	char addr3[]="192.16.1";
	char addr[16];
	char *str=NULL,*str2=NULL;

	int err = 0;

	/* 测试函数inet_aton */
	err = inet_aton(addr1, &ip);
	if(err){
		printf("inet_aton:string %s value is:0x%x\n",addr1, ip.s_addr);
	}else{
		printf("inet_aton:string %s  error\n",addr1);
	}

	/* inet_addr,先测试192.168.1.1,在测试255.255.255.255 */
	ip.s_addr = inet_addr(addr1);
	if(err != -1){
		printf("inet_addr:string %s value is:0x%x\n",addr1, ip.s_addr);
	}else{
		printf("inet_addr:string %s  error\n",addr1);
	};
	ip.s_addr = inet_addr(addr2);
	if(ip.s_addr != -1){
		printf("inet_addr:string %s value is:0x%x\n",addr2, ip.s_addr);
	}else{
		printf("inet_addr:string %s  error\n",addr2);
	};

	/* inet_ntoa,先测试192.168.1.1,在测试255.255.255.255
	*	证明函数的不可重入性
	 */
	ip.s_addr = 192<<24|168<<16|1<<8|1;
	str = inet_ntoa(ip);
	ip.s_addr = 255<<24|255<<16|255<<8|255;
	str2 = inet_ntoa(ip);
	printf("inet_ntoa:ip:0x%x string1 %s,pre is:%s \n",ip.s_addr,str2,str);

	/* 测试函数inet_addr */
	ip.s_addr = inet_addr(addr3);
	if(err != -1){
		printf("inet_addr:string %s value is:0x%x\n",addr3, ip.s_addr);
	}else{
		printf("inet_addr:string %s  error\n",addr3);
	};
	str = inet_ntoa(ip);
	printf("inet_ntoa:string %s ip:0x%x \n",str,ip.s_addr);

	/* 测试函数inet_lnaof,获得本机地址 */
	inet_aton(addr1, &ip);
	local.s_addr = htonl(ip.s_addr);
	local.s_addr = inet_lnaof(ip);
	str = inet_ntoa(local);
	printf("inet_lnaof:string %s ip:0x%x \n",str,local.s_addr);

	/* 测试函数inet_netof,获得本机地址 */
	network.s_addr = inet_netof(ip);
	printf("inet_netof:value:0x%x \n",network.s_addr);

	return 0;
}

运行结果:

二.安全的地址转换函数inet_pton(),inet_ntop()

函数inet_pton(),inet_ntop()都是可以重入的,同时支持多种地址类型,包含IPV4和IPV6。

函数介绍:

下面用一段代码介绍使用方法:

#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>

#define ADDRLEN 16
int main(int argc, char *argv[])
{
	struct in_addr ip;
	char IPSTR[]="192.168.1.1";		/* 网络地址字符串 */
	char addr[ADDRLEN];				/* 保存转换后的字符串IP地址,16个字节大小 */
	const char*str=NULL;
	int err = 0;					/* 返回值 */

	/* 测试函数inet_pton转换192.168.1.1为二进制形式 */
	err = inet_pton(AF_INET, IPSTR, &ip);	/* 将字符串转换为二进制 */
	if(err > 0){
		printf("inet_pton:ip,%s value is:0x%x\n",IPSTR,ip.s_addr);
	}

	/* 测试函数inet_ntop转换192.168.1.1为字符串 */
	ip.s_addr = htonl(192<<24|168<<16|12<<8|255);/*192.168.12.255*/
	/*将二进制网络字节序192.168.12.255转换为字符串*/
	str = (const char*)inet_ntop(AF_INET, (void*)&ip, (char*)&addr[0], ADDRLEN);
	if(str){
		printf("inet_ntop:ip,0x%x is %s\n",ip.s_addr,str);
	}

	return 0;
}

运行结果:

inet_pton:ip,192.168.1.1 value is:0x101a8c0

inet_ntop:ip,0xff0ca8c0 is 192.168.12.255

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-18 03:14:00

Linux网络编程--字符串IP和二进制IP转换的相关文章

linux网络编程基础--(转自网络)

转自 http://www.cnblogs.com/MyLove-Summer/p/5215287.html Linux下的网络编程指的是socket套接字编程,入门比较简单. 1. socket套接字介绍 socket机制其实就是包括socket, bind, listen, connect, accept等函数的方法,其通过指定的函数实现不同的协议(IP4,IP6等)的数据在不同层之间的传输和获取等处理.其实个人理解socket就是处于应用层和TCP/IP协议之间的一个中间层,具体的数据分析

linux c 网络编程:用域名获取IP地址或者用IP获取域名 网络地址转换成整型 主机字符顺序与网络字节顺序的转换

用域名获取IP地址或者用IP获取域名 #include<stdio.h> #include<sys/socket.h> #include<netdb.h> int main(int argc,char **aggv) { struct hostent *host; char hostname[]="www.163.com"; char hostname2[]="www.baidu.com"; struct in_addr in;

很全的linux网络编程技巧

注:作者王晓,本人认为总结得很好,故记之,绝无侵权之意. 1. LINUX网络编程基础知识 1 1.1. TCP/IP协议概述 1 1.2. OSI参考模型及TCP/IP参考模型 1 1.3. TCP协议 3 1.4. UDP协议 5 1.5. 协议的选择 6 2. 网络相关概念 6 2.1. socket概念 7 2.2. socket类型 8 2.3. socket信息数据结构 8 2.4. 数据存储优先顺序的转换 8 2.5. 地址格式转化 9 2.6. 名字地址转化 10 3. sock

Linux网络编程的一般步骤(1)

一.套接字的地址结构. IPV4套接字地址结构通常也称为"网际套接字地址结构",它以sockaddr_in 命名;POSIX定义如下: #include <stdio.h> struct in_addr { unsigned long s_addr; /*32-bit IPv4 address network byte ordered*/ }; /*socketaddr结构体一般是一般是用作参数来使用,指明地址信息*/ struct sockaddr { unsigned

Linux网络编程入门 (转载)

http://www.cnblogs.com/RascallySnake/archive/2012/01/04/2312564.html (一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端         网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户端        在网络程序中,如果一个程序主动和外面的程序通信,那么我们把这个程序称为客户端程序. 比如我们使用ftp程序从另外一        个地方获取文件

Linux网络编程(二)

服务套和客户机的信息函数 1.字节转换函数 在网络上面有着许多类型的机器,这些机器在表示数据的字节顺序是不同的, 比如i386芯片是低字节在内存地址的低端,高字节在高端,而alpha芯片却相反. 为了统一起来,在Linux下面,有专门的字节转换函数. unsigned long int htonl(unsigned long int hostlong)     unsigned short int htons(unisgned short int hostshort)     unsigned

嵌入式 Linux网络编程(一)——Socket网络编程基础

嵌入式 Linux网络编程一--Socket网络编程基础 一.Socket简介 1.网络中进程间通信 本机进程使用进程号区别不同的进程进程间通信方式有管道.信号.消息队列.共享内存.信号量等.网络中进程间的通信首先需要识别进程所在主机在网络中的唯一标识即网络层的IP地址主机上的进程可以通过传输层的协议与端口号识别. 2.Socket原理 Socket是应用层与TCP/IP协议族通信的中间软件抽象层是一种编程接口.Socket屏蔽了不同网络协议的差异支持面向连接(Transmission Cont

Linux网络编程入门

(一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍 客户端和服务端 网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户端 在网络程序中,如果一个程序主动和外面的程序通信,那么我们把这个程序称为客户端程序. 比如我们使用ftp程序从另外一 个地方获取文件的时候,是我们的ftp程序主动同外面进行通信(获取文件), 所以这个地方我们的ftp程序就是客户端程序. 服务端 和客户端相对应的程序即为服务端程序.被动的等待外面的程序来和自己通

Linux网络编程函数

转自:http://blog.csdn.net/hrbeuwhw/article/details/8050911 1.字节序函数 #include<netinet.h> uint16_t htons(uint16_t host16bitvalue); uint32_t htonl(uint32_t host32bitvalue); 返回:网络字节序值 uint16_t ntohs(uint16_t net16bitvalue); uint32_t ntohl(uint32_t net32bit