连续调用inet_ntoa打印出错的问题

本文通过示例说明了inet_ntoa函数在多处调用时出现的问题,指出该函数未保证线程安全,并可能导致打印结果错误。具体表现为在连续调用时,后一次调用会覆盖前一次调用的返回地址。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

近日写程序,在打印信息的时候调用了inet_ntoa函数,出现了打印一直出错的情况。google了一下,是因为inet_ntoa这类函数没有保证线程安全,其实现原理是在静态内容中申请一块内存,每次调用后返回该静态内存的指针,若是在同一个printf语句中连续调用两次inet_ntoa函数会导致后调用的覆盖先覆盖的那个。

 

举个例子(参考了某位前辈的blogs):

 1 #include <stdio.h>
 2 #include <stdlib.h>
 3 #include <netinet/in.h>
 4 int main()
 5 {
 6         struct sockaddr_in ra = {0};
 7         struct sockaddr_in sa = {0};
 8         ra.sin_addr.s_addr = -217732928; //192.168.5.243    
 9         sa.sin_addr.s_addr = -939415360; //192.168.5.200  
10         printf("ip of recv:%s\nip of send:%s\n",\
11                         inet_ntoa(ra.sin_addr),inet_ntoa(sa.sin_addr));
12         printf("ip of send:%s\n",inet_ntoa(sa.sin_addr));
13         printf("ip of recv:%s\n",inet_ntoa(ra.sin_addr));
14 }

执行结果:

1 [root@host-b lab]# gcc test_ntoa.c 
2 [root@host-b lab]# ./a.out 
3 ip of recv:192.168.5.243
4 ip of send:192.168.5.243
5 ip of send:192.168.1.200
6 ip of recv:192.168.5.243

结果的3、4两行对应code中的11行,结果的5、6两行对用code中的12、13两行,3、4两行的结果明显不正确,原因就是在inet_ntoa(ra.sin_addr)返回的地址将inet_ntoa(sa.sin_addr)的返回地址

转载于:https://www.cnblogs.com/lovemyspring/p/3330040.html

实现用vs打开#include <iostream> #include <string> #include <ctime> #include <winsock2.h> // Windows套接字库 #pragma comment(lib, "ws2_32.lib") // 链接套接字库 #include <Ws2tcpip.h> // 添加这个头文件 int main() { WSADATA wsaData; SOCKET serverSocket, clientSocket; sockaddr_in serverAddr, clientAddr; int clientAddrLen = sizeof(clientAddr); char buffer[1024] = { 0 }; // 初始化 Winsock if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) { std::cerr << "WSAStartup failed" << std::endl; return 1; } // 创建套接字 serverSocket = socket(AF_INET, SOCK_STREAM, 0); if (serverSocket == INVALID_SOCKET) { std::cerr << "Socket creation failed" << std::endl; WSACleanup(); return 1; } // 配置服务器地址 serverAddr.sin_family = AF_INET; serverAddr.sin_addr.s_addr = INADDR_ANY; // 监听所有可用接口 serverAddr.sin_port = htons(8080); // 端口号 // 绑定套接字 if (bind(serverSocket, (sockaddr*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) { std::cerr << "Bind failed" << std::endl; closesocket(serverSocket); WSACleanup(); return 1; } // 开始监听 if (listen(serverSocket, 1) == SOCKET_ERROR) { std::cerr << "Listen failed" << std::endl; closesocket(serverSocket); WSACleanup(); return 1; } std::cout << "服务器已启动,等待客户端连接..." << std::endl; // 接受客户端连接 clientSocket = accept(serverSocket, (sockaddr*)&clientAddr, &clientAddrLen); if (clientSocket == INVALID_SOCKET) { std::cerr << "Accept failed" << std::endl; closesocket(serverSocket); WSACleanup(); return 1; } // 使用 inet_ntop 替代 inet_ntoa char clientIP[INET_ADDRSTRLEN]; inet_ntop(AF_INET, &clientAddr.sin_addr, clientIP, INET_ADDRSTRLEN); std::cout << "客户端 " << clientIP << ":" << ntohs(clientAddr.sin_port) << " 已连接" << std::endl; try { while (true) { // 接收客户端消息 int bytesReceived
05-11
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值