求助,关于socket和多线程的通信,照着孙鑫的教材改编成线程后逻辑错误,急急急

猴头 2010-12-14 03:02:06
代码如下

这是服务器的代码:
#include <Winsock2.h>
#include <stdio.h>
#include <string>
using std::string;
#include <iostream>
using namespace std;

DWORD WINAPI FuncServer(LPVOID lp);
int connectPotr;

void main()
{

printf("《主服务器》已开启,请等待..........\n\n");
//加载套接字库
WORD wVersionRequested;
WSADATA wsaData;
int err;
wVersionRequested=MAKEWORD(1,1);

err=WSAStartup( wVersionRequested , &wsaData );
if(err!=0)
{
return ;
}

if (LOBYTE(wsaData.wVersion)!=1||HIBYTE(wsaData.wVersion)!=1)
{
WSACleanup();
return ;
}

//创建用于监听的套接字
SOCKET sockSrv=socket(AF_INET,SOCK_STREAM,0);
SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=htonl(INADDR_ANY);
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(6000);
//6000是端口号
//绑定套接字
bind(sockSrv,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));
//将套接字设为监听模式,准备接受客户请求
listen(sockSrv,5);
SOCKADDR_IN addrClient;
int len=sizeof(SOCKADDR);
int count_Client=1;
while (count_Client)
{

//等待客户请求到来;
SOCKET sockConn=accept(sockSrv,(SOCKADDR*)&addrClient,&len);
//创建新的套接字,用此套接字与客户进行链接


char sendBuf[100];

printf("主服务器正在等待客户请求............\n\n");
sprintf(sendBuf,"Welcome 客户: %s to servers <猴子>\n\n",inet_ntoa(addrClient.sin_addr));

printf("客户端地址:%s\n\n", inet_ntoa(addrClient.sin_addr));
send( sockConn , sendBuf , strlen(sendBuf)+1 , 0);
char recvBuf[100];
//接收数据
recv(sockConn,recvBuf,100,0);

//打印接收的数据
printf("从TcpClient接收到的数据是:%s\n\n",recvBuf);

cout<<"在主函数内\n";
//关闭套接字
closesocket(sockConn);
printf("接收到第 %d 个客户.....\n\n",count_Client);
count_Client++;
connectPotr++;

HANDLE hThreadl;
hThreadl=CreateThread(NULL,0,FuncServer,NULL,0,NULL);
Sleep(1000);
CloseHandle(hThreadl);
connectPotr++;

}
}


DWORD WINAPI FuncServer(LPVOID lp)
{
printf("《子服务器》已开启,请等待..........\n\n");
//加载套接字库
WORD wVersionRequested;
WSADATA wsaData;
int err;
wVersionRequested=MAKEWORD(1,1);

err=WSAStartup( wVersionRequested , &wsaData );
if(err!=0)
{
return 0;
}

if (LOBYTE(wsaData.wVersion)!=1||HIBYTE(wsaData.wVersion)!=1)
{
WSACleanup();
return 0;
}

//创建用于监听的套接字
SOCKET sockSrv=socket(AF_INET,SOCK_STREAM,0);

SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=htonl(INADDR_ANY);
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(6010);
//6010是端口号

//绑定套接字
bind(sockSrv,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));

//将套接字设为监听模式,准备接受客户请求
listen(sockSrv,5);

SOCKADDR_IN addrClient;
int len=sizeof(SOCKADDR);
int count_Client=1;
while (count_Client)
{

//等待客户请求到来;
SOCKET sockConn=accept(sockSrv,(SOCKADDR*)&addrClient,&len);
//创建新的套接字,用此套接字与客户进行链接


char sendBuf[100]={0};

printf("服务器正在等待客户请求............\n\n");
sprintf(sendBuf,"Welcome 客户: %s to servers <猴子子服务器>\n\n",inet_ntoa(addrClient.sin_addr));

printf("客户端地址:%s\n\n", inet_ntoa(addrClient.sin_addr));
//(addrClient.sin_addr)获得的就是客户端的地址
//可已通过接收到的客户端的ip地址来获取对应的地区编码
//发送数据
send( sockConn , sendBuf , strlen(sendBuf)+1 , 0);

char recvBuf[100]={0};
//接收数据
recv(sockConn,recvBuf,100,0);
//打印接收的数据
if(0!=recvBuf[0])
printf("从TcpClient接收到的数据是:%s\n\n",recvBuf);
//关闭套接字
cout<<"在子函数内\n";
system("pause");/////****&&&&&&**&&^%%%^&*^%$%^&*&^%$%^&

closesocket(sockConn);
printf("接收到第 %d 条数据.....\n\n",count_Client);
count_Client++;

}

return 0;

}

客户端的代码:
#include <WINSOCK2.H>
#include <stdio.h>
#include <stdexcept>
#include <string>
using std::string;

#include <iostream>
using namespace std;

int connectPort;

void connectSrv(int connectPort);

void main()
{
printf("《客户端》正在连接\n\n");
//加载套接字
WORD wVersionRequestd;
WSADATA wsaData;
int err;

wVersionRequestd=MAKEWORD(1,1);

err=WSAStartup(wVersionRequestd,&wsaData );
if (err!=0)
{
return ;
}

if (LOBYTE( wsaData.wVersion )!=1||HIBYTE( wsaData.wVersion )!=1)
{
WSACleanup();
return ;
}

//创建套接字
SOCKET sockClient = socket(AF_INET,SOCK_STREAM ,0);

SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=inet_addr("127.0.0.1");
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(6000);
//向服务器发起请求
printf("向服务器发起链接请求........\n\n");
connect(sockClient,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));
send(sockClient,"My name is 小猴子!!!",strlen("My name is 小猴子!!!")+1,0);

//接收数据
char recvBuf[100]={0};
try{
recv(sockClient,recvBuf,100,0);
if(0==recvBuf[0])
{
throw runtime_error("ERROR!未连接到服务器!");
}
//while(1)
//{
SOCKET sockClient = socket(AF_INET,SOCK_STREAM ,0);
SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=inet_addr("127.0.0.1");
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(6000);
connect(sockClient,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));
//发送一条数据后得重新发起连接

//printf("客户端准备接受数据..........\n\n");
printf("客户端接收到的数据是:%s\n\n",recvBuf);
//发送数据
//printf("客户端准备发送数据...........\n\n");

//string Str;
//cin>>Str;
//char *p=(char*)Str.c_str();
send(sockClient,"OK",strlen("OK")+1,0);

//};
closesocket(sockClient);
WSACleanup();

cout<<"\n马上建立子连接....."<<endl<<endl;
connectSrv( connectPort );


//关闭套接字
}
catch(runtime_error& err)
{
cout<<err.what()<<endl<<endl;
closesocket(sockClient);
WSACleanup();
}




system("pause");
return ;

}

void connectSrv(int Port)
{
Sleep(2000);
int connectPort=Port;
printf("《客户端》准备发送数据........\n\n");
//加载套接字
WORD wVersionRequestd;
WSADATA wsaData;
int err;

wVersionRequestd=MAKEWORD(1,1);

err=WSAStartup(wVersionRequestd,&wsaData );
if (err!=0)
{
return ;
}

if (LOBYTE( wsaData.wVersion )!=1||HIBYTE( wsaData.wVersion )!=1)
{
WSACleanup();
return ;
}

//创建套接字
SOCKET sockClient = socket(AF_INET,SOCK_STREAM ,0);

SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=inet_addr("127.0.0.1");
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(6010);
//向服务器发起请求
printf("向服务器发起链接请求........\n\n");
connect(sockClient,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));
send(sockClient,"My name is 小猴子2222!!!",strlen("My name is 小猴子2222!!!")+1,0);

//接收数据
char recvBuf[100]={0};
try{
recv(sockClient,recvBuf,100,0);
if(0==recvBuf[0])
{
throw runtime_error("ERROR!未连接到服务器!");
}
while(1)
{
SOCKET sockClient = socket(AF_INET,SOCK_STREAM ,0);
SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=inet_addr("127.0.0.1");
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(6010);
//connect(sockClient,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));
//发送一条数据后得重新发起连接

printf("客户端准备接受数据..........\n\n");
printf("客户端接收到的数据是:%s\n\n",recvBuf);
//发送数据
printf("客户端准备发送数据...........\n\n");

string Str;
cin>>Str;
char *p=(char*)Str.c_str();
send(sockClient,p,strlen(p)+1,0);

};

//关闭套接字
}
catch(runtime_error& err)
{
cout<<err.what()<<endl<<endl;
}
closesocket(sockClient);
WSACleanup();

system("pause");
return ;


}


我想让客户端和服务器建立起连接后,让服务器创建一个线程,让子线程用端口6010和客户端进行通讯,原意是这样的,但是运行时候却出现了以下错误:
客户端可以发送数据,但是服务器确接收不到,还陷入了死循环。
是哪的逻辑错误了吗???还是结构就不对???
请各位大侠帮忙修改修改,
谢谢各位
用控制台写的每用界面。
...全文
108 12 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
12 条回复
切换为时间正序
请发表友善的回复…
发表回复
猴头 2010-12-15
  • 打赏
  • 举报
回复
把accept穿进去之后,不用另外申请端口,就可以实现多个客户端和服务器对话了
是这样啊
我以为还得重新申请端口呢
[Quote=引用 4 楼 visualeleven 的回复:]
你服务器端accept一个连接以后,创建线程,将accept返回的SOCKET作为线程参数传递进去就可以了
[/Quote]
猴头 2010-12-14
  • 打赏
  • 举报
回复
不是联两次服务器,比如说,第一次联上之后用的是端口6000,但在服务器上这个端口还会有别的客户端去连接,我想让第一个客户端用6000端口连上之后,让服务器再建一个端口假如是6010,然后让客户端通过这个端口和服务器创建的子线程通信。
[Quote=引用 10 楼 visualeleven 的回复:]
引用 9 楼 yan_hyz 的回复:010
但是 我觉得问题不在这里啊,
我在主函数里面断开了以前的链接,在线程中重新建立的新的连接,
既然建立新的连接不得再次accept吗???
还有就是
recv(sockConn,recvBuf,100,0);
//打印接收的数据
if(0!=recvBuf[0])
printf("从TcpClient接收到的数据是:%s\n\n",recvBu……
[/Quote]
Eleven 2010-12-14
  • 打赏
  • 举报
回复
[Quote=引用 9 楼 yan_hyz 的回复:]
但是 我觉得问题不在这里啊,
我在主函数里面断开了以前的链接,在线程中重新建立的新的连接,
既然建立新的连接不得再次accept吗???
还有就是
recv(sockConn,recvBuf,100,0);
//打印接收的数据
if(0!=recvBuf[0])
printf("从TcpClient接收到的数据是:%s\n\n",recvBuf);
//关闭套接字
……
[/Quote]
你的意思是一个客户端要connect 2次服务器?你不觉得逻辑上存在问题吗?
猴头 2010-12-14
  • 打赏
  • 举报
回复
但是 我觉得问题不在这里啊,
我在主函数里面断开了以前的链接,在线程中重新建立的新的连接,
既然建立新的连接不得再次accept吗???
还有就是
recv(sockConn,recvBuf,100,0);
//打印接收的数据
if(0!=recvBuf[0])
printf("从TcpClient接收到的数据是:%s\n\n",recvBuf);
//关闭套接字
被跳过不执行的问题[Quote=引用 8 楼 visualeleven 的回复:]
引用 7 楼 yan_hyz 的回复:
通过那个函数传进去啊??

引用 4 楼 visualeleven 的回复:
你服务器端accept一个连接以后,创建线程,将accept返回的SOCKET作为线程参数传递进去就可以了

创建线程的函数啊,就是你上面写的CreateThread,不过一般不建议使用这个,而是使用_beginthreadex()
[/Quote]
Eleven 2010-12-14
  • 打赏
  • 举报
回复
[Quote=引用 7 楼 yan_hyz 的回复:]
通过那个函数传进去啊??

引用 4 楼 visualeleven 的回复:
你服务器端accept一个连接以后,创建线程,将accept返回的SOCKET作为线程参数传递进去就可以了
[/Quote]
创建线程的函数啊,就是你上面写的CreateThread,不过一般不建议使用这个,而是使用_beginthreadex()
猴头 2010-12-14
  • 打赏
  • 举报
回复
通过那个函数传进去啊??[Quote=引用 4 楼 visualeleven 的回复:]
你服务器端accept一个连接以后,创建线程,将accept返回的SOCKET作为线程参数传递进去就可以了
[/Quote]
猴头 2010-12-14
  • 打赏
  • 举报
回复
但是 还有一个问题,就是,死循环的问题 system("pause");/////****&&&&&&**&&^%%%^&*^%$%^&*&^%$%^&
执行之前接收到了数据,但是这句去掉之后,他直接跳过接收数据的语句进入了死循环
[Quote=引用 4 楼 visualeleven 的回复:]
你服务器端accept一个连接以后,创建线程,将accept返回的SOCKET作为线程参数传递进去就可以了
[/Quote]
猴头 2010-12-14
  • 打赏
  • 举报
回复
这位仁兄,这个该怎么传呢??类型不一样,我强制转换,他还报错。。。
[Quote=引用 2 楼 ouyh12345 的回复:]
楼主在线程里,又做了accept的操作
应该把accept后的socket作为变量传到线程里,然后再用此socket来接收和发送
[/Quote]
Eleven 2010-12-14
  • 打赏
  • 举报
回复
你服务器端accept一个连接以后,创建线程,将accept返回的SOCKET作为线程参数传递进去就可以了
haoruixiang 2010-12-14
  • 打赏
  • 举报
回复
[Quote=引用 2 楼 ouyh12345 的回复:]
楼主在线程里,又做了accept的操作
应该把accept后的socket作为变量传到线程里,然后再用此socket来接收和发送
[/Quote]

最好把指针传进去
ouyh12345 2010-12-14
  • 打赏
  • 举报
回复
楼主在线程里,又做了accept的操作
应该把accept后的socket作为变量传到线程里,然后再用此socket来接收和发送
猴头 2010-12-14
  • 打赏
  • 举报
回复
在服务器端创建的子线程函数里面:
recv(sockConn,recvBuf,100,0);没有正确接收到数据,好像跳过了这一句直接进入和下一次循环

18,363

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC 网络编程
c++c语言开发语言 技术论坛(原bbs)
社区管理员
  • 网络编程
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧