C++ WinSock 2 - 使用 getpeername() 函数时套接字崩溃
Posted
技术标签:
【中文标题】C++ WinSock 2 - 使用 getpeername() 函数时套接字崩溃【英文标题】:C++ WinSock 2 - socket crashing when getpeername() function is used 【发布时间】:2015-08-11 16:16:38 【问题描述】:每当我使用函数getpeername()
时,我的程序在调试时崩溃并返回255。它printf
是主机名,当它开始尝试printf
对等名称时,它会显示“对等名称:”,然后在它告诉我对等名称之前崩溃。我知道这个程序没有多大意义。我只是在弄乱一些 WinSock 2 函数,看看它们都做了什么。
我的代码:
#include <windows.h>
#include <winsock2.h>
#include <conio.h>
#include <stdio.h>
#include <iostream>
using namespace std;
#define SCK_VERSION2 0x0202
#define DEFAULT_BUFLEN 512
#define DEFAULT_PORT 27015
int main()
//----------------------
// Declare and initialize variables.
WSADATA wsaData;
int iResult;
SOCKET ConnectSocket = INVALID_SOCKET;
struct sockaddr_in clientService;
char name[500] = "";
sockaddr sName;
int sNameSize = sizeof(sName);
char *sendbuf = "Test";
char recvbuf[DEFAULT_BUFLEN];
int recvbuflen = DEFAULT_BUFLEN;
//----------------------
// Initialize Winsock
iResult = WSAStartup(MAKEWORD(2,2), &wsaData);
if (iResult != NO_ERROR)
printf("WSAStartup failed: %d\n", iResult);
return 1;
//----------------------
// Create a SOCKET for connecting to server
ConnectSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (ConnectSocket == INVALID_SOCKET)
printf("Error at socket(): %i\n", WSAGetLastError() );
WSACleanup();
return 1;
//----------------------
// The sockaddr_in structure specifies the address family,
// IP address, and port of the server to be connected to.
clientService.sin_family = AF_INET;
clientService.sin_addr.s_addr = inet_addr( "127.0.0.1" );
clientService.sin_port = htons(444);
//----------------------
// Connect to server.
iResult = connect( ConnectSocket, (SOCKADDR*) &clientService, sizeof(clientService) );
if ( iResult == SOCKET_ERROR)
closesocket (ConnectSocket);
printf("Unable to connect to server: %i\n", WSAGetLastError());
WSACleanup();
return 1;
//----------------------
//Get local host name
iResult = gethostname(name, sizeof(name));
if (iResult == NO_ERROR)
printf("Host Name: %s\n", name);
else if (iResult == SOCKET_ERROR)
printf("Could not resolve host name: %i", WSAGetLastError());
iResult = getpeername(ConnectSocket, (struct sockaddr*)&sName, &sNameSize);
if (iResult == NO_ERROR)
printf("Peer Name: %s", sName);
else if (iResult == SOCKET_ERROR)
printf("Could not get peer name: %i", WSAGetLastError());
// Send an initial buffer
iResult = send( ConnectSocket, sendbuf, (int)strlen(sendbuf), 0 );
if (iResult == SOCKET_ERROR)
printf("send failed: %d\n", WSAGetLastError());
closesocket(ConnectSocket);
WSACleanup();
return 1;
printf("Bytes Sent: %i\n", iResult);
// shutdown the connection since no more data will be sent
iResult = shutdown(ConnectSocket, SD_SEND);
if (iResult == SOCKET_ERROR)
printf("shutdown failed: %d\n", WSAGetLastError());
closesocket(ConnectSocket);
WSACleanup();
return 1;
// Receive until the peer closes the connection
do
iResult = recv(ConnectSocket, recvbuf, recvbuflen, 0);
if ( iResult > 0 )
printf("Bytes received: %s\n", recvbuf); //printf("Bytes received: %d\n", iResult);
else if ( iResult == 0 )
printf("Connection closed\n");
else
printf("recv failed: %d\n", WSAGetLastError());
while( iResult > 0 );
// cleanup
closesocket(ConnectSocket);
WSACleanup();
system("PAUSE");
return 0;
【问题讨论】:
你期望这会做什么?printf("Peer Name: %s", sName);
?您的代码将 sName
声明为 sockaddr,而不是以 null 结尾的 C 字符串。
【参考方案1】:
试试这个:
结构 sockaddr_in sockaddr; int namelen = sizeof(sockaddr); if(!getpeername(socket, (struct sockaddr *)&sockaddr, &namelen)) printf("对端名称:%s\n", inet_ntoa((in_addr)(* (in_addr*)&sockaddr.sin_addr.S_un.S_addr)));【讨论】:
【参考方案2】:你的电话有两个问题。
您的套接字是 IPv4 套接字,因此 getpeername()
需要 sockaddr_in
,就像 connect()
一样。但是您已将 sName
变量声明为 sockaddr
。您需要为套接字类型使用正确的结构类型:
sockaddr_in sName;
您正试图打印出sName
,就好像它是一个以空字符结尾的字符串一样,但事实并非如此。它是一个二进制结构。您必须将其内容转换为以 null 结尾的字符串,例如 inet_ntoa()
:
iResult = getpeername(ConnectSocket, (struct sockaddr*)&sName, &sNameSize);
if (iResult == 0)
printf("Peer Name: %s", inet_ntoa(sName.sin_addr));
else
printf("Could not get peer name: %i", WSAGetLastError());
【讨论】:
peer name 应该是一堆数字吗? @EdwardSeverinsen:是的。getpeername()
返回对等方的 IP 地址。您确实了解 IP 地址实际上是什么,对吗?一组数字表示设备在网络中的地址。您要做的是打印出该 IP 地址的 人类可读 表示,这就是为什么您不能直接打印出 sockaddr_in
结构的内容,而是必须将其转换为首先是可打印格式。
Nvm 我忘了把%d
换成%s
XD以上是关于C++ WinSock 2 - 使用 getpeername() 函数时套接字崩溃的主要内容,如果未能解决你的问题,请参考以下文章
如何使用 C++ 在 Winsock 中将大字符串从客户端发送到服务器