inet_ntoa()返回字符串的生命周期

在做linux下面的网络编程时写了如下一段程序

 

[cpp:showcolumns]  view plain copy
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
  1. //(省去N行……)  
  2. char *source,*dest;  
  3. struct in_addr saddr;  
  4. struct in_addr daddr;  
  5. saddr.s_addr (in_addr_t)cmd.source.v_uint;  //cmd.source.v_uint装载了由字符串通过inet_aton()转化成的源IP的网络字节,这里无错,不必深究:)   
  6. daddr.s_addr (in_addr_t)cmd.dest.v_uint;    //同上  
  7.   
  8. source (char*)inet_ntoa(saddr);  
  9. dest (char*)inet_ntoa(daddr);  
  10.   
  11. printf(“source[%s]  dest[%s]”,source,dest);  
  12. //(省去N行……)  

执行之后打印结果总是源IP和目的IP一样,如下(程序名为pro):
#./pro -s 192.1.1.1 -d 192.2.2.2
#source[192.2.2.2] dest[192.2.2.2]

             检查了很久都没发现任何错误,最始还以为赋值时不小心把源IP赋成了目的IP,但是事实上编写的时候还是比较细心, 没犯这种错误,呵呵:)。
             最后一步步排查,将saddr.s_addr与daddr.s_addr的数值都打印出来了,发现二者的数值完全不一样,既然它们两不 一样,通过inet_ntoa()返回的字符串也应该不一样,所以,问题就出在了inet_ntoa()的应用上了。man了一下,发现了对inet_ntoa的这么一行说明:

The inet_ntoa() function converts the Internet host address in, given
in network byte order, to a string in IPv4 dotted-decimal notation.
The string is returned in a statically allocated buffer, which subse-
quent calls will overwrite.


       红色字体的意思是inet_ntoa()返回的字符串是临时装在一个静态分配的缓冲区里面,下一次调用此函数的时候缓冲区会被重写 。哈哈,终于发现为什么转化为字符串后,源地址老是与目的地址一样了,关键原因在于指针的使用,前面是将用来装载源地址和目的地址字符串的source与dest声明成了指针,那么,赋值之后二者指针就会都指向inet_ntoa()的临时缓冲区里,在给dest赋值时,缓冲区被重写了,所以source指针的值就变成了目的地址,二者自然就一样了。因此,改成下面的代码就没问题了:

 

[cpp:showcolumns]  view plain copy
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
  1.    //(依然省去N行……)   
  2.       char source[16],dest[16];   //声明成数组,可以将内容从缓冲区里复制过来,面不是用指针指向缓冲区  
  3. int saddr_len,daddr_len;  
  4. struct in_addr saddr;  
  5. struct in_addr daddr;  
  6.   
  7. saddr.s_addr (in_addr_t)cmd.source.v_uint;  
  8. daddr.s_addr (in_addr_t)cmd.dest.v_uint;      
  9. saddr_len strlen(inet_ntoa(saddr));  
  10. daddr_len strlen(inet_ntoa(daddr));  
  11.   
  12. if(saddr_len <=15 && daddr_len <=15)     //IP址的字符串最多为15位  
  13.  
  14.     memcpy(source,inet_ntoa(saddr),saddr_len);  
  15.     memcpy(dest,inet_ntoa(daddr),daddr_len);  
  16.     source[saddr_len] '/0' 
  17.     dest[daddr_len] '/0' 
  18.   
  19. }else 
  20.     printf("ADDR Error/n");  
  21.     exit(1);  
  22.  
  23. printf(“source[%s]  dest[%s]”,source,dest);  
  24.   
  25. //(依然省去N行……)  

习惯了用家用电器不看说明书-_-!,看来,以后在使用东西之前还是多看看说明,少走弯路



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值