Socket中常见的几个转换函数(htonl,htons,ntohl,ntohs,inet_addr,inet_ntoa)

本文介绍Socket编程中IP地址转换函数的使用方法,包括inet_addr()用于将点分十进制IP地址转换为网络字节顺序的无符号长整型,以及inet_ntoa()用于将网络字节顺序的IP地址转换回点分十进制格式。
Socket中常见的几个转换函数(htonl,htons,ntohl,ntohs,inet_addr,inet_ntoa) 
2009年12月27日 
  htonl() htons() ntohl() ntohs()及inet_ntoa() inet_addr()的用法 
  注:其中的h表示“host”,n表示“net”,l表示“long”, 
                      s表示“short”a表示“ascii”,ddr表示“in_addr结构体” 
  现在我们很幸运,因为我们有很多的函数来方便地操作 IP 地址。没有 必要用手工计算它们,也没有必要用"一个sockaddr_in结构体变量ina,你有一个IP地址"132.241.5.10" 要储存在其中,你就要用到函数inet_addr(),将IP地址从点数格式转换成无符号长整型。使用方法如下: 
  ina.sin_addr.s_addr = inet_addr("132.241.5.10"); 
  注意,inet_addr()返回的地址已经是网络字节格式,所以你无需再调用 函数htonl()。 
  现在你可以将IP地址转换成长整型了。有没有其相反的方法呢? 它可以将一个in_addr结构体输出成点数格式?这样的话,你就要用到函数 inet_ntoa()("ntoa"的含义是"network to ascii"),就像这样: 
  printf("%s",inet_ntoa(ina.sin_addr)); 
  它将输出IP地址。需要注意的是inet_ntoa()将结构体in-addr作为一 个参数,不是长整形。同样需要注意的是它返回的是一个指向一个字符的指针。它是一个由inet_ntoa()控制的静态的固定的指针,所以每次调用 inet_ntoa(),它就将覆盖上次调用时所得的IP地址。例如: 
  char *a1, *a2; 
  a1 = inet_ntoa(ina1.sin_addr); /* 这是198.92.129.1 */ 
  a2 = inet_ntoa(ina2.sin_addr); /* 这是132.241.5.10 */ 
  printf("address 1: %s ",a1); 
  printf("address 2: %s ",a2); 
  输出如下: 
  address 1: 132.241.5.10 
  address 2: 132.241.5.10 
  内存结构: 
   
   
   
  假如你需要保存这个IP地址,使用strcpy()函数来指向你自己的字符指针。 
  ================================================================== 
  htonl()表示将32位的主机字节顺序转化为32位的网络字节顺序 htons()表示将16位的主机字节顺序转化为16位的网络字节顺序(ip地址是32位的端口号是16位的 ) 
  inet_ntoa() 
  简述: 
  将网络地址转换成“.”点隔的字符串格式。 
  #include 
  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套接口调用前复制出来。 
  参见: 
  inet_addr(). 
  测试代码如下 
  include 
  #include 
  #include 
  #include 
  #include 
  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里面的可变参数的求值是从右到左的,仅此而已。 

C语言:我通过抓包发现没有循环arp请求,请帮我看看怎么回事,并修改,只需要使其能够扫描即可: // ARP条目结构 typedef struct arp_entry { struct in_addr ip; // IP地址 struct ether_addr mac; // MAC地址 time_t last_update; // 最后更新时间戳 struct arp_entry *next; // 链表指针 } arp_entry_t; // 扫描器配置 typedef struct arp_scanner { int enabled; // 功能开关 (1=启用, 0=禁用) time_t scan_cycle; // 扫描周期() time_t entry_ttl; // 条目有效期() int packet_interval; // 发包间隔(毫秒) struct in_addr ip_start;// 起始IP struct in_addr ip_end; // 结束IP arp_entry_t *arp_table; // ARP表头指针 pthread_mutex_t lock; // 线程安全锁 } arp_scanner_t; // 全局配置实例 arp_scanner_t scanner; // 构造ARP请求,todo:好像不会发出arp请求,得确定一下 void build_arp_request(struct ether_header *ehdr, struct ether_arp *arp_req, struct in_addr target_ip, struct ether_addr local_mac) { // 以太网帧头 memset(ehdr->ether_dhost, 0xFF, ETHER_ADDR_LEN); // 广播地址 memcpy(arp_req->arp_sha, &local_mac, ETHER_ADDR_LEN); ehdr->ether_type = htons(ETHERTYPE_ARP); // ARP请求体 arp_req->arp_hrd = htons(ARPHRD_ETHER); // 硬件类型 arp_req->arp_pro = htons(ETHERTYPE_IP); // 协议类型 arp_req->arp_hln = ETHER_ADDR_LEN; // 硬件地址长度 arp_req->arp_pln = sizeof(in_addr_t); // 协议地址长度 arp_req->arp_op = htons(ARPOP_REQUEST); // 操作码=请求 memcpy(arp_req->arp_sha, &local_mac, ETHER_ADDR_LEN); memset(arp_req->arp_tha, 0x00, ETHER_ADDR_LEN); // 目标MAC初始为空 memcpy(arp_req->arp_spa, &scanner.ip_start, sizeof(in_addr_t)); // 发送方IP memcpy(arp_req->arp_tpa, &target_ip, sizeof(in_addr_t)); // 目标IP } // 扫描线程函数 void *scan_thread(void *arg) { struct ether_addr local_mac; if (get_local_mac("ens33", &local_mac) < 0) { // 替换为实际网卡名 perror("获取MAC失败"); return NULL; } // 创建原始套接字 int sockfd = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ARP)); if (sockfd < 0) { perror("socket创建失败"); return NULL; } while (scanner.enabled) { struct in_addr current_ip = scanner.ip_start; // 遍历IP范围 while (current_ip.s_addr <= scanner.ip_end.s_addr) { char packet[sizeof(struct ether_header) + sizeof(struct ether_arp)]; struct ether_header *ehdr = (struct ether_header *)packet; struct ether_arp *arp_req = (struct ether_arp *)(packet + sizeof(struct ether_header)); build_arp_request(ehdr, arp_req, current_ip, local_mac); // 发送ARP请求 if (sendto(sockfd, packet, sizeof(packet), 0, NULL, 0) < 0) { } // IP递增 current_ip.s_addr = htonl(ntohl(current_ip.s_addr) + 1); usleep(scanner.packet_interval * 1); // 发包间隔 } sleep(scanner.scan_cycle); // 等待扫描周期 } close(sockfd); return NULL; } // ARP响应处理 void handle_arp_reply(unsigned char *buffer) { struct ether_arp *arp_res = (struct ether_arp *)(buffer + sizeof(struct ether_header)); // 只处理ARP响应 if (ntohs(arp_res->arp_op) != ARPOP_REPLY) return; struct in_addr src_ip; memcpy(&src_ip, arp_res->arp_spa, sizeof(src_ip)); struct ether_addr src_mac; memcpy(&src_mac, arp_res->arp_sha, ETHER_ADDR_LEN); // 检查IP是否在扫描范围内 if (src_ip.s_addr < scanner.ip_start.s_addr || src_ip.s_addr > scanner.ip_end.s_addr) return; // 更新ARP表 arp_entry_t *entry = find_arp_entry(src_ip); if (entry) { entry->mac = src_mac; entry->last_update = time(NULL); } else { add_arp_entry(src_ip, src_mac); } } // 监听线程函数 void *listen_thread(void *arg) { int sockfd = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ARP)); if (sockfd < 0) { perror("监听socket失败"); return NULL; } unsigned char buffer[ETH_FRAME_LEN]; while (scanner.enabled) { ssize_t len = recvfrom(sockfd, buffer, sizeof(buffer), 0, NULL, NULL); if (len < (ssize_t)(sizeof(struct ether_header) + sizeof(struct ether_arp))) { continue; } handle_arp_reply(buffer); } close(sockfd); return NULL; }
最新发布
08-27
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值