【问题标题】:Server cannot receive the clients message in c winsock programmingcwinsock编程中服务器无法接收客户端消息
【发布时间】:2015-11-28 19:19:17
【问题描述】:

你好朋友我是csocket编程的新手,我在网上看了一些教程并开始工作。下面的程序是我在WINDOWS中使用winsock2库的socket客户端和服务器程序。

每当我运行程序时,在服务器接受客户端连接之前没有错误。但是在客户端发送消息后,服务器无法接收消息。 WSAGetLastError() 函数返回错误代码为 10038。我尝试更改端口,增加和减少消息大小。 朋友们帮帮我。

服务器代码

#include<stdio.h>
#include<conio.h>
#include<WinSock2.h>
#pragma comment(lib,"ws2_32.lib")
int _tmain(int argc, _TCHAR* argv[])
{
WSADATA wsa;
int r;
char buf[4];
WSAStartup(MAKEWORD(2,2),&wsa);

SOCKET s= socket(AF_INET,SOCK_STREAM,0);

if ( s==INVALID_SOCKET)
{
    printf("\n failed to create socket error code : %d ",WSAGetLastError());
    WSACleanup();
    return 1;
}

sockaddr_in sin,cl;
sin.sin_family=AF_INET;
sin.sin_port=htons(80);
sin.sin_addr.s_addr=INADDR_ANY;

if(bind (s,(sockaddr *)&sin,sizeof(sin))==SOCKET_ERROR)
{
    printf("\nSocket not Bound\n");
    WSACleanup(); 
    return 1;
}

r=listen(s,2);
if(r==SOCKET_ERROR)
{
    printf("\nListening Failed\n");
    WSACleanup();
    return 1;
}
printf("\n Listening\n");

SOCKET client;
r= sizeof(sockaddr_in);
printf("\n Ready to accept");
while (client=accept(s,(sockaddr *)&cl,&r)!= INVALID_SOCKET)
{
printf("\nNew client found\n\n");
if(client == SOCKET_ERROR)
{
    printf("\nError connecting client error code : %d ",WSAGetLastError());
    WSACleanup();
    return 1;
}

r=recv(client,buf,4,0);
if(r==SOCKET_ERROR)
{
    printf("\nSocket not connected error code : %d",WSAGetLastError());
    WSACleanup();
    return 1;
}

printf("%s\n",buf);
}
getchar();
return 0;
}

客户端代码

#include <stdio.h>
#include <conio.h>
#include <WinSock2.h>
#pragma comment(lib,"ws2_32.lib")

int _tmain(int argc, _TCHAR* argv[])
{

WSADATA wsa;
WSAStartup(MAKEWORD(2,2),&wsa);
int r;
char *buf="data";

SOCKET t=socket(AF_INET,SOCK_STREAM,0);
if(t==SOCKET_ERROR)
{
    printf("\nFailed socket error code : %d ",WSAGetLastError());
    WSACleanup();
    return 1;
}

sockaddr_in server;
server.sin_family=AF_INET;
server.sin_addr.s_addr=inet_addr("192.168.0.142");
server.sin_port=htons(80);

printf("\nConnecting\n");
r=connect(t,(sockaddr *)&server,sizeof(server));
if(r==SOCKET_ERROR)
{
    printf("\nFailed to connect error code : %d ",WSAGetLastError());
    WSACleanup();
    return 1;
}
printf("\n connected\n");
 
 printf("\nSending data\n");
 r=send(t,buf,strlen(buf),0);
 if (r==SOCKET_ERROR)
 {
     printf("\nFailed to connect error code : %d ",WSAGetLastError());
     WSACleanup();
     return 1;
 }
 printf("\n Data sent\n");
return 0;
}

【问题讨论】:

  • 您必须注意的一件事是,C 中的字符串是以空值结尾的,这意味着在标记结束位置的字符串之后有一个符号 \0。所以"data"strlen() 是4,因为它有4 个字符,但它的总大小实际上是5 个字节。您也必须发送\0,否则服务器端的buf 将不是有效的C 字符串,当您尝试printf() 时可能会导致未定义的行为。
  • 另外,如果不指定协议,我不知道你会得到什么样的行为。在这种情况下,socket() 似乎应该接收IPPROTO_TCP 作为第三个参数,而不是0
  • ZeroMQ 值得一看。这是一种比处理原始套接字更容易使用网络连接的方法。查看zeromq.org/distro:microsoft-windows 进行下载,并搜索“ZeroMQ 指南”。

标签: c sockets winsock winsock2


【解决方案1】:

首先我们看看 10038 是什么意思。

SAENOTSOCK
10038 (0x2736)
An operation was attempted on something that is not a socket.

好的,所以你认为你有一个套接字,但你真的没有在你的一个电话中。你不是很具体的 WSAGetLastError() 函数返回 10038,但我会猜测这个调用之后的调用:

while (client=accept(s,(sockaddr *)&cl,&r)!= INVALID_SOCKET)

在 C 中,首先处理 !=,然后是赋值。因此,您最终会将accept 返回的套接字与INVALID_SOCKET 进行比较,然后将该布尔值分配给client。您希望分配套接字然后进行比较,因此您应该在分配周围使用括号:

while ((client = accept(s,(sockaddr *)&cl,&r)) != INVALID_SOCKET)

此外,socket() 返回 INVALID_SOCKET 而不是 SOCKET_ERROR,尽管混淆两者不会巧合。

此外,如果要打印,则需要在接收方以空值终止字符串。

char buf[5];  /* 4 characters + 1 null-terminator */
...
buf[sizeof(buf) - 1] = '\0';    /* enforce that the buffer is always null terminated regardless of what the sender sends. */

【讨论】:

  • 感谢个人的帮助。现在代码工作得很好。但是我为保持连接而添加的while循环仍然只执行一次。你能帮我解决这个问题吗
  • 您必须更具体。在我看来,您似乎没有一个 while 循环来使连接保持在您发布的代码中。如果您需要一些新代码方面的帮助,您应该发布一个新问题。
猜你喜欢
  • 2013-03-28
  • 1970-01-01
  • 1970-01-01
  • 2012-12-03
  • 1970-01-01
  • 1970-01-01
  • 2012-04-07
  • 1970-01-01
  • 2016-05-27
相关资源
最近更新 更多