所以我正在尝试创建一个迷你聊天应用程序,只是为了进入并理解套接字编程和关于线程的一两件事,我正在使用c ++ Mfc,我可以在客户端和之间建立连接服务器(用netstat验证它)但是对于send和recv函数,我不太明白应该怎么做,我从客户端应用程序发送消息但服务器似乎没有收到它
服务器的源代码:
int RcvThread();
SOCKET s;
void CChat_ServerDlg::OnBnClickedButton2()
{
WSADATA w;
int error = WSAStartup ( 0x0202,&w);
if(error)
{
OnCancel();
}
if (w.wVersion != 0x0202)
{
WSACleanup ();
OnCancel();
}
SOCKADDR_IN addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(DEFAULT_PORT);
addr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
s = socket ( AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (s == INVALID_SOCKET)
{
OnCancel();
}
if (bind(s, (LPSOCKADDR)&addr, sizeof(addr)) == SOCKET_ERROR)
{
OnCancel();
}
listen (s, SOMAXCONN);
CreateThread(NULL, NULL, (LPTHREAD_START_ROUTINE) RcvThread, NULL, NULL, NULL);
int buffsize = 1024;
char msg[1024] = "a";
int marker;
}
int RcvThread()
{
char sbuffer[256];
char buffer[sizeof(sbuffer)] = {0};
for(;; )
{
if(recv(s, buffer, sizeof(sbuffer), NULL) > 0)
{
memcpy(&sbuffer, buffer, sizeof(sbuffer));
MessageBox(hnd,sbuffer,"message",NULL);
}
}
return 0;
}
客户端的源代码:
SOCKET s;
void CChat_ClientDlg::OnBnClickedOk()
{
WSADATA wsadata;
int error = WSAStartup(0x0202,&wsadata);
if (error)
{
MessageBox("Error","ERRR");
OnCancel();
}
if (wsadata.wVersion != 0x0202)
{
WSACleanup ();
OnCancel();
}
SOCKADDR_IN target;
target.sin_family = AF_INET;
target.sin_port = htons(3124);
target.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
s = socket ( AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (s == INVALID_SOCKET)
{
OnCancel();
}
if (connect(s, (SOCKADDR *)&target, sizeof(target)) == SOCKET_ERROR)
{
OnCancel();
}
}
按钮发送消息:
void CChat_ClientDlg::OnBnClickedButton2()
{
char* Msg = new char[256];
Msg = "abdouabdouabdou";
send(s,Msg,256,NULL);
}
答案 0 :(得分:1)
您缺少套接字的基本概念。 listen套接字(代码中的s)用于接受连接请求。它不用于发送/接收数据。你必须为此创建另一个套接字。您需要更多的文档和样本研究。一个来源是:
http://msdn.microsoft.com/en-us/library/windows/desktop/ms738545(v=vs.85).aspx
此外,您的线程不正确。如果使用CreateThread,则线程函数必须与为CreateThread指定的签名匹配。但是在MFC应用程序中,您应该使用AfxBeginThread而不是CreateThread。
答案 1 :(得分:0)
首先,相信send()和recv()将发送和接收数据包是一种常见的误解。如果您使用TCP(您应该用于聊天程序),recv()可以返回介于0和您指定的缓冲区限制之间的任意数量的字节。因此,如果您发送256个字节,则您的recv()函数可以将其拆分为消息的两个,三个或更多个片段,或者它可以将第一个消息的结尾与第二个消息的开头一起返回。经常做的是消息的第一个n字节(n取决于你的最大消息大小是多少)标记消息的长度。由于您将协议限制为256字节消息长度,因此1个字节就足够了。收到第一个字节后,分配一个与您的消息一样大的消息缓冲区,并将recv()放入...... well ...接收循环,直到它收到整个消息。你需要将偏移量与接收缓冲区等进行兼顾。
其次,你还不太明白bind()/ listen()/ accept()系统是如何工作的。 listen()函数将套接字设置为被动模式,在该模式下,它正在侦听新客户端。 accept()最终建立与新客户端的连接,并返回一个 new 套接字,然后用于与客户端通信。原始(listen)套接字继续监听新客户端。
第三,你的send函数处理你的char-buffers不正确。我根据我的建议纠正了一下(消息的第一个字节标记了以下消息的长度):
void CChat_ClientDlg::OnBnClickedButton2()
{
std::string myMessage = "abdouabdouabdou";
unsigned char buffer[256];
buffer[0] = (unsigned char)myMessage.length();
memcpy(buffer+1, myMessage.c_str(), std::min(myMessage.length(), 255));
send(s, buffer, 256, NULL);
}
即使该方法也是不正确的,因为send()返回的发送字节数可能小于我的缓冲区大小,尽管这种情况很少见。