为什么我的c++ Winsock服务器无法从客户端接收数据
Why is my c++ Winsock server unable to receive data from the client?
我尝试运行一个简单的客户端和服务器程序。然后,我尝试从客户机向服务器发送一条消息,并让服务器回显该消息。问题是,每次我这样做,我得到一个10057错误。我也不知道为什么。
服务器端代码
#include <stdio.h>
#include <winsock2.h>
#include <stdlib.h>
#define MYBUFSIZE 128
void shutdown_close(SOCKET s)
{
// Tell the operating system the socket is
// about to be closed
shutdown(s, SD_BOTH);
// close the socket….
closesocket(s);
}
void shutdown_close_and_exit(SOCKET s)
{
shutdown_close(s);
exit(1);
}
void main(int argc, char *argv[])
{
SOCKET srvr_socket;
struct sockaddr_in srvr_addr;
struct sockaddr_in clnt_addr; // Client address
int addr_len = 0;
WSADATA wsaData;
char recv_buf[MYBUFSIZE];
int recv_msg_len;
short portnum;
if (argc == 2)
{
portnum = atoi(argv[1]);
printf("Setting port number to %dn", portnum);
} else
{
fprintf(stderr, "Usage: %s port_num_to_listen_atn",
argv[0]);
exit(1);
}
// Init Winsock
if (WSAStartup(MAKEWORD(2, 0), &wsaData) != 0)
{
fprintf(stderr, "Error: WSAStartup() failed");
exit(1);
}
// Create UDP datagram socket for incoming connections
if ((srvr_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == SOCKET_ERROR)
{
fprintf(stderr, "Error: socket() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(srvr_socket);
}
// Construct local address structure
memset(&srvr_addr, 0, sizeof(srvr_addr));
srvr_addr.sin_family = AF_INET;
srvr_addr.sin_addr.s_addr = inet_addr("127.0.0.1"); // Accept messages on any network interface.
srvr_addr.sin_port = htons(portnum); // Bind the port number specified on the command line.
/* Bind to the local address */
if (bind(srvr_socket, (struct sockaddr *) &srvr_addr, sizeof(srvr_addr)) == SOCKET_ERROR)
{
fprintf(stderr, "Error: socket() failed with error %d", WSAGetLastError());
shutdown_close_and_exit(srvr_socket);
}
listen(srvr_socket, SOMAXCONN);
// Loop forever
for (;;)
{
// Set the size of the in-out parameter, where the client address
// and port number will be stored by the OS
addr_len = sizeof(clnt_addr);
if(accept(srvr_socket,(SOCKADDR *) &srvr_addr, &addr_len) == SOCKET_ERROR)
{
fprintf(stderr, "Error: accept() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(srvr_socket);
}
// Receive message from client
if ((recv_msg_len = recvfrom(srvr_socket, recv_buf, MYBUFSIZE, 0,
(struct sockaddr *) &clnt_addr, &addr_len )) == SOCKET_ERROR) {
fprintf(stderr, "Error: recvfrom() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(srvr_socket);
} else {
printf("Received message of size %d: %sn from client %s:%dn",
recv_msg_len, recv_buf, inet_ntoa(clnt_addr.sin_addr),
ntohs(clnt_addr.sin_port));
}
// 'echo' message back to client
if ((recv_msg_len = sendto(srvr_socket, recv_buf, recv_msg_len, 0,
(struct sockaddr *) &clnt_addr, addr_len )) == SOCKET_ERROR) {
fprintf(stderr, "Error: sendto() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(srvr_socket);
} else {
printf("Echo'd message %s to client %s:%dn",
recv_buf, inet_ntoa(clnt_addr.sin_addr),
ntohs(clnt_addr.sin_port));
}
}
}
,下面是客户端代码:
#include <stdio.h>
#include <winsock2.h>
#include <stdlib.h>
#define BUFSIZE 128 /* Size of receive buffer */
void shutdown_close(SOCKET s)
{
// Tell the operating system the socket is
// about to be closed
shutdown(s, SD_BOTH);
// close the socket….
closesocket(s);
}
void shutdown_close_and_exit(SOCKET s)
{
shutdown_close(s);
exit(1);
}
void main(int argc, char *argv[])
{
SOCKET servSock;
struct sockaddr_in srvr_addr;
struct sockaddr_in recv_addr;
int addr_len = 0;
WSADATA wsaData;
char mesg_buf[BUFSIZE];
char *mesg_to_send;
long server_IP;
short portnum;
if (argc == 4)
{
portnum = atoi(argv[1]);
printf("Setting port number to %dn", portnum);
server_IP = inet_addr(argv[2]);
printf("Target server IP address is %sn", argv[2]);
mesg_to_send = argv[3];
} else
{
fprintf(stderr, "Usage: %s server_port_num server_IP_address message_to_sendn",
argv[0]);
exit(1);
}
// Init Winsock
if (WSAStartup(MAKEWORD(2, 0), &wsaData) != 0)
{
fprintf(stderr, "Error: WSAStartup() failed with error %dn", WSAGetLastError());
exit(1);
}
// Create socket for incoming connections
if ((servSock = socket(AF_INET,SOCK_STREAM, IPPROTO_TCP)) == SOCKET_ERROR)
{
fprintf(stderr, "Error: socket() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(servSock);
}
// Construct local address structure
// with server address. This is like
// addressing the envelope of a letter.
memset(&srvr_addr, 0, sizeof(srvr_addr));
srvr_addr.sin_family = AF_INET;
srvr_addr.sin_addr.s_addr = server_IP;
srvr_addr.sin_port = htons(portnum);
if ( connect(servSock, (SOCKADDR*) &srvr_addr, sizeof(srvr_addr)) == SOCKET_ERROR)
{
fprintf(stderr, "Error: connect() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(servSock);
}
// Set the size of the in-out parameter
addr_len = sizeof(recv_addr);
if (sendto(servSock, mesg_to_send, strlen(mesg_to_send) + 1, 0,
(sockaddr *) &srvr_addr, addr_len) == SOCKET_ERROR)
{
fprintf(stderr, "Error: sendto() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(servSock);
}
printf("Send message %s to server at %s:%dn", argv[3], argv[2], portnum);
// Sleep 1 full second to allow message to get to server and be sent back...
Sleep(1000);
addr_len = sizeof(recv_addr);
if (recvfrom(servSock, mesg_buf, BUFSIZE, 0,
(sockaddr *) &recv_addr, &addr_len) == SOCKET_ERROR)
{
fprintf(stderr, "Error: recvfrom() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(servSock);
}
printf("Received message %s from %s:%dn", mesg_buf,
inet_ntoa(recv_addr.sin_addr), ntohs(recv_addr.sin_port));
// close socket gracefully
shutdown_close(servSock);
}
有人能看出代码有什么问题吗?
您应该为SOCK_STREAM
套接字使用recv
,而不是recvfrom
。
(但这是一大堆代码,没有明显的故障指示,所以可能还有其他问题)
用send()
代替sendto()
,用recv()
代替recvfrom()
。另外,您在服务器端将错误的sockaddr_in
传递给accept()
。
我明白是怎么回事了。我的服务器从未建立连接,因为我从未与accept函数建立连接。换句话说,我接受了一种联系,但不记得它。我应该创建一个套接字来接受连接,并将其设置为accept函数,如下所示:
Socket acceptSocket;
...
AcceptSocket = accept(srvr_socket,(SOCKADDR *) &clnt_addr, &addr_len) ;
if(AcceptSocket == INVALID_SOCKET)
{
fprintf(stderr, "Error: accept() failed with error %dn", WSAGetLastError());
shutdown_close_and_exit(srvr_socket);
}
我也应该使用recv()和send()而不是sendto()和recvfrom()像RichieHindle提到的,因为它们是为UDP无连接套接字而不是TCP连接套接字设计的
当我从recvfrom()接收错误10057时,我的初始代码段如下:
if (listen(s, SOMAXCONN) == SOCKET_ERROR) {
printf("Error in Listening .. ErroR Code: %d n", WSAGetLastError());
exit(EXIT_FAILURE);
}
else
printf("Listeningn");
//Accept and incoming connection
puts("Waiting for incoming connections...");
c = sizeof(struct sockaddr_in);
new_socket = accept(s, (struct sockaddr *)&client, &c);
if (new_socket == INVALID_SOCKET)
{
printf("accept failed with error code : %d", WSAGetLastError());
return 1;
}
else
puts("Connection accepted");
recv_size = recvfrom(s , client_msg, sizeof(client_msg), 0,
(SOCKADDR *)&recv_from, &addrlen);
我正在尝试从我正在监听的套接字上的客户端接收数据。"s"而不是新的套接字,我得到了客户端连接接受,即。"new_socket"。
After making the following changes:
recv_size = recvfrom(new_socket , client_msg, sizeof(client_msg), 0, (SOCKADDR
*)&recv_from, &addrlen);
我可以完美地从客户端应用程序接收数据。
相关文章:
- "unknown ca"自生成的 CA、证书和客户端/服务器
- 如何将函数集合传递给客户端类,以便将它们当作客户端类本身的成员使用
- 使用调试/崩溃报告将应用程序部署到客户端
- 如何在本地机器上运行c++和javascript客户端代码(hackerbank风格)
- 如何通过套接字将文本文件的内容从服务器发送到客户端
- 从服务器传输到客户端的消息不会出现
- OpenSSL TLS服务器-使用客户端证书白名单
- 当服务中的事件被触发时,如何将响应从服务发送回客户端?
- 我可以与 python 服务器而不是 c++ 客户端建立 tcp/ip 套接字吗?
- 提升 Asio TCP 服务器 处理多个客户端
- boost::asio UDP 广播客户端仅接收"fast"数据包
- 如何绑定 C++ gRPC 客户端的网络接口
- C++套接字客户端到 Python 服务器未创建连接
- 用于解析 win64 堆栈跟踪的命令行客户端(可以访问符号服务器)
- 将相机数据从服务器实时流式传输到客户端
- 如何将 Firebase 与基于 Linux 的客户端应用配合使用,以便与服务器进行双向消息通信
- GRPC C++ TLS 客户端 grpc::SslCredentials() 方法不返回
- C 套接字服务器不接受任何客户端 /客户端无法连接
- TCP 服务器/客户端:客户端 recv() 返回空白缓冲区
- Visual Studio 2015 C++ 应用程序需要在客户端客户端上使用 api-ms-win-crt-runti