一个socket连接错误困扰我2天了,请大家帮帮忙,小菜鸟先谢谢大家了,socket错误10054!

sese53 2009-02-26 02:41:32
这个例子是我照着孙鑫的那本《VC++深入详解》书的网络编程的一个聊天程序的例子,是一个基于对话框的MFC程序,已经运行成功。我向自己127.0.0.1发送数据,可是总接受不到数据。调试后在这一个函数中退出。socket错误代码为10054。


DWORD WINAPI C网络聊天室程序实现Dlg::RecvProc(LPVOID lpParameter)
{
SOCKET sock = ((RECVPARAM*)lpParameter)->sock;
HWND hwnd = ((RECVPARAM*)lpParameter)->hwnd;
delete lpParameter;

SOCKADDR_IN addrFrom;
int len = sizeof(SOCKADDR);

char recvBuf[200];
char tempBuf[300];
int retval;

while (true)
{
retval = recvfrom(sock, recvBuf, 200, 0, (SOCKADDR*)&addrFrom, &len);
if (SOCKET_ERROR == retval)//程序进入这个分支后,break;没有接受到下面的代码,retval为10054;
{
char str[10];
int error = WSAGetLastError();
CString my("错误代码为:");
AfxMessageBox(my + itoa(error, str, 10));
break;
}
sprintf(tempBuf, "%s说:%s", inet_ntoa(addrFrom.sin_addr), recvBuf);
::PostMessage(hwnd, WM_RECVDATA, 0, (LPARAM)tempBuf);
}

return 0;
}

希望大家帮忙解决一下,没有做好这个例子真的是太慌了。下面我把所有主要代码贴出来。


//// 网络聊天室程序实现Dlg.cpp : 实现文件
//

#include "stdafx.h"
#include "网络聊天室程序实现.h"
#include "网络聊天室程序实现Dlg.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif


// C网络聊天室程序实现Dlg 对话框




C网络聊天室程序实现Dlg::C网络聊天室程序实现Dlg(CWnd* pParent /*=NULL*/)
: CDialog(C网络聊天室程序实现Dlg::IDD, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void C网络聊天室程序实现Dlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(C网络聊天室程序实现Dlg, CDialog)
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
//}}AFX_MSG_MAP
ON_MESSAGE(WM_RECVDATA, OnRecvData)
ON_BN_CLICKED(IDC_BTN_SEND, &C网络聊天室程序实现Dlg::OnBtnSend)
END_MESSAGE_MAP()


// C网络聊天室程序实现Dlg 消息处理程序

BOOL C网络聊天室程序实现Dlg::OnInitDialog()
{
CDialog::OnInitDialog();

// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
// 执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标

// TODO: 在此添加额外的初始化代码
InitSocket();
RECVPARAM *pRecvParam = new RECVPARAM;
pRecvParam->sock = m_socket;
pRecvParam->hwnd = m_hWnd;

HANDLE hThread = CreateThread(NULL, 0, RecvProc, (LPVOID)pRecvParam, 0, NULL);
CloseHandle(hThread);

return TRUE; // 除非将焦点设置到控件,否则返回 TRUE
}

// 如果向对话框添加最小化按钮,则需要下面的代码
// 来绘制该图标。对于使用文档/视图模型的 MFC 应用程序,
// 这将由框架自动完成。

void C网络聊天室程序实现Dlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用于绘制的设备上下文

SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);

// 使图标在工作矩形中居中
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;

// 绘制图标
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialog::OnPaint();
}
}

//当用户拖动最小化窗口时系统调用此函数取得光标显示。
//
HCURSOR C网络聊天室程序实现Dlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_hIcon);
}


BOOL C网络聊天室程序实现Dlg::InitSocket(void)
{
m_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (INVALID_SOCKET == m_socket)
{
MessageBox("套接字创建失败");
return FALSE;
}

SOCKADDR_IN addrSock;
addrSock.sin_family = AF_INET;
addrSock.sin_port = htons(6000);
addrSock.sin_addr.S_un.S_addr = htonl(INADDR_ANY);

int retval;

retval = bind(m_socket, (SOCKADDR*)&addrSock, sizeof(SOCKADDR));

if (SOCKET_ERROR == retval)
{
closesocket(m_socket);
MessageBox("绑定套接字失败");
return FALSE;
}

return true;
}

DWORD WINAPI C网络聊天室程序实现Dlg::RecvProc(LPVOID lpParameter)
{
SOCKET sock = ((RECVPARAM*)lpParameter)->sock;
HWND hwnd = ((RECVPARAM*)lpParameter)->hwnd;
delete lpParameter;

SOCKADDR_IN addrFrom;
int len = sizeof(SOCKADDR);

char recvBuf[200];
char tempBuf[300];
int retval;

while (true)
{
retval = recvfrom(sock, recvBuf, 200, 0, (SOCKADDR*)&addrFrom, &len);
if (SOCKET_ERROR == retval)
{
char str[10];
int error = WSAGetLastError();
CString my("错误代码为:");
AfxMessageBox(my + itoa(error, str, 10));
break;
}
sprintf(tempBuf, "%s说:%s", inet_ntoa(addrFrom.sin_addr), recvBuf);
::PostMessage(hwnd, WM_RECVDATA, 0, (LPARAM)tempBuf);
}

return 0;
}

LRESULT C网络聊天室程序实现Dlg::OnRecvData(WPARAM wParam, LPARAM lParam)
{
CString str = (char*)lParam;
CString strTemp;
GetDlgItemText(IDC_EDIT_RECV, strTemp);
str += "\r\n";
str += strTemp;
SetDlgItemText(IDC_EDIT_RECV, str);
return 0;
}

void C网络聊天室程序实现Dlg::OnBtnSend()
{
DWORD dwIP;
((CIPAddressCtrl*)GetDlgItem(IDC_IPADDRESS1))->GetAddress(dwIP);

SOCKADDR_IN addrTo;
addrTo.sin_family = AF_INET;
addrTo.sin_port = 5666;
addrTo.sin_addr.S_un.S_addr = htonl(dwIP);

CString strSend;
GetDlgItemText(IDC_EDIT_SEND, strSend);

sendto(m_socket, strSend, strSend.GetLength()+1, 0, (SOCKADDR*)&addrTo, sizeof(SOCKADDR));
SetDlgItemText(IDC_EDIT_SEND,"");
}


// 网络聊天室程序实现.cpp : 定义应用程序的类行为。
//

#include "stdafx.h"
#include "网络聊天室程序实现.h"
#include "网络聊天室程序实现Dlg.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif


// C网络聊天室程序实现App

BEGIN_MESSAGE_MAP(C网络聊天室程序实现App, CWinApp)
ON_COMMAND(ID_HELP, &CWinApp::OnHelp)
END_MESSAGE_MAP()


// C网络聊天室程序实现App 构造

C网络聊天室程序实现App::C网络聊天室程序实现App()
{
// TODO: 在此处添加构造代码,
// 将所有重要的初始化放置在 InitInstance 中
}


// 唯一的一个 C网络聊天室程序实现App 对象

C网络聊天室程序实现App theApp;


// C网络聊天室程序实现App 初始化

BOOL C网络聊天室程序实现App::InitInstance()
{
if (!AfxSocketInit())
{
AfxMessageBox("加载套接字库失败!");
return FALSE;
}

// 如果一个运行在 Windows XP 上的应用程序清单指定要
// 使用 ComCtl32.dll 版本 6 或更高版本来启用可视化方式,
//则需要 InitCommonControlsEx()。否则,将无法创建窗口。
INITCOMMONCONTROLSEX InitCtrls;
InitCtrls.dwSize = sizeof(InitCtrls);
// 将它设置为包括所有要在应用程序中使用的
// 公共控件类。
InitCtrls.dwICC = ICC_WIN95_CLASSES;
InitCommonControlsEx(&InitCtrls);

CWinApp::InitInstance();

AfxEnableControlContainer();

// 标准初始化
// 如果未使用这些功能并希望减小
// 最终可执行文件的大小,则应移除下列
// 不需要的特定初始化例程
// 更改用于存储设置的注册表项
// TODO: 应适当修改该字符串,
// 例如修改为公司或组织名
SetRegistryKey(_T("应用程序向导生成的本地应用程序"));

C网络聊天室程序实现Dlg dlg;
m_pMainWnd = &dlg;
INT_PTR nResponse = dlg.DoModal();
if (nResponse == IDOK)
{
// TODO: 在此处放置处理何时用“确定”来关闭
// 对话框的代码
}
else if (nResponse == IDCANCEL)
{
// TODO: 在此放置处理何时用“取消”来关闭
// 对话框的代码
}

// 由于对话框已关闭,所以将返回 FALSE 以便退出应用程序,
// 而不是启动应用程序的消息泵。
return FALSE;
}


// 网络聊天室程序实现Dlg.h : 头文件
//

#pragma once
#define WM_RECVDATA WM_USER + 1

struct RECVPARAM
{
SOCKET sock;
HWND hwnd;
};

// C网络聊天室程序实现Dlg 对话框
class C网络聊天室程序实现Dlg : public CDialog
{
// 构造
public:
C网络聊天室程序实现Dlg(CWnd* pParent = NULL); // 标准构造函数

// 对话框数据
enum { IDD = IDD_MY_DIALOG };

protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持


// 实现
protected:
HICON m_hIcon;
SOCKET m_socket;

// 生成的消息映射函数
virtual BOOL OnInitDialog();
afx_msg void OnPaint();
afx_msg HCURSOR OnQueryDragIcon();
afx_msg LRESULT OnRecvData(WPARAM wParam, LPARAM lParam);
DECLARE_MESSAGE_MAP()
public:
BOOL InitSocket(void);
static DWORD WINAPI RecvProc(LPVOID lpParameter);
public:
afx_msg void OnBtnSend();
};

// 网络聊天室程序实现.h : PROJECT_NAME 应用程序的主头文件
//

#pragma once

#ifndef __AFXWIN_H__
#error "在包含此文件之前包含“stdafx.h”以生成 PCH 文件"
#endif

#include "resource.h" // 主符号


// C网络聊天室程序实现App:
// 有关此类的实现,请参阅 网络聊天室程序实现.cpp
//

class C网络聊天室程序实现App : public CWinApp
{
public:
C网络聊天室程序实现App();

// 重写
public:
virtual BOOL InitInstance();

// 实现

DECLARE_MESSAGE_MAP()
};

extern C网络聊天室程序实现App theApp;

...全文
6138 21 打赏 收藏 转发到动态 举报
写回复
用AI写文章
21 条回复
切换为时间正序
请发表友善的回复…
发表回复
瓶盒 2009-02-28
  • 打赏
  • 举报
回复
[Quote=引用 16 楼 sese53 的回复:]
引用 15 楼 bottlebox 的回复:
htons(6000)=6000 ???
你这个是什么意思呢?
[/Quote]
用你的代码调试了一下,改成addrTo.sin_port =htons(6000)后一切正常
opens555 2009-02-28
  • 打赏
  • 举报
回复
帮顶
sese53 2009-02-28
  • 打赏
  • 举报
回复
[Quote=引用 17 楼 prodiving 的回复:]
首先应该是你发送数据的目的地址是无效的,和你OnBtnSend中的设置端口有问题(注意htons),
至于线程为什么10054出错退出,是因为你OnBtnSend里发送用的sock句柄和接收的是一致的,发送了无效地址,导致socket出问题。

你要确定你OnBtnSend中发送的目的地址是有效的,

顺便说下,UDP的应用目的地址有时确实是不能确定有效的,而导致socket出问题后不能向其他地址(即使有效)再发送数据了, 很久前这是winsock的一个b…
[/Quote]
我给你发了谈话了。希望你能看到。
sese53 2009-02-27
  • 打赏
  • 举报
回复
[Quote=引用 12 楼 prodiving 的回复:]
你这个是udp的,不能那telnet测试,你这个程序打开两份, 一个发一个接收,那个收的线程一般是不会退出的了,

我看了代码,想了想,估计你那个接收的线程的socket出现这个错误是你只开一份程序, 然后用这个程序bind在6000后用来接收的socket也拿来发送数据?然后发送到无效的5666端口,结果这个socket出现异常,那个收的线程才退出了
[/Quote]
朋友,这个程序是既是接收,也是发送啊。都绑定了6000端口,如果开2个,就会报错,绑定套接字失败。
而且书上也是一个程序自己给自己发。
sese53 2009-02-27
  • 打赏
  • 举报
回复
[Quote=引用 11 楼 bottlebox 的回复:]
C/C++ code
void C网络聊天室程序实现Dlg::OnBtnSend()
{
DWORD dwIP;
((CIPAddressCtrl*)GetDlgItem(IDC_IPADDRESS1))->GetAddress(dwIP);

SOCKADDR_IN addrTo;
addrTo.sin_family = AF_INET;
addrTo.sin_port =htons(6000);//这里的问题,监听的是6000号端口,当然是向这个端口发数据
addrTo.sin_addr.S_un.S_addr = htonl(dwIP);
。。。。
}
[/Quote]
朋友,我知道你说的什么了。但是其实接受和绑定到一个端口的。我只是改动了,发错了一份而已,我2个都是6000.
addrSock.sin_port = htons(6000);
addrTo.sin_port = 6000;
prodiving 2009-02-27
  • 打赏
  • 举报
回复
首先应该是你发送数据的目的地址是无效的,和你OnBtnSend中的设置端口有问题(注意htons),
至于线程为什么10054出错退出,是因为你OnBtnSend里发送用的sock句柄和接收的是一致的,发送了无效地址,导致socket出问题。

你要确定你OnBtnSend中发送的目的地址是有效的,

顺便说下,UDP的应用目的地址有时确实是不能确定有效的,而导致socket出问题后不能向其他地址(即使有效)再发送数据了, 很久前这是winsock的一个bug,不过后来ms提供另一个接口:
//
//Winsock 忽略 ICMP 端口无法访问控制消息 http://support.microsoft.com/kb/245442/
//http://support.microsoft.com/kb/263823/
//解决udp发往一个无效地址后10054错误,导致无法向其他地址发送数据
BOOL bNewBehavior = FALSE;
DWORD dwBytesReturned = 0;
WSAIoctl(s,
SIO_UDP_CONNRESET,
&bNewBehavior,
sizeof bNewBehavior,
NULL,
0,
&dwBytesReturned,
NULL,
NULL);
//
sese53 2009-02-27
  • 打赏
  • 举报
回复
[Quote=引用 15 楼 bottlebox 的回复:]
htons(6000)=6000 ???
[/Quote]
你这个是什么意思呢?
prodiving 2009-02-27
  • 打赏
  • 举报
回复
你这个是udp的,不能那telnet测试,你这个程序打开两份, 一个发一个接收,那个收的线程一般是不会退出的了,

我看了代码,想了想,估计你那个接收的线程的socket出现这个错误是你只开一份程序, 然后用这个程序bind在6000后用来接收的socket也拿来发送数据?然后发送到无效的5666端口,结果这个socket出现异常,那个收的线程才退出了
瓶盒 2009-02-27
  • 打赏
  • 举报
回复
htons(6000)=6000 ???
瓶盒 2009-02-27
  • 打赏
  • 举报
回复

void C网络聊天室程序实现Dlg::OnBtnSend()
{
DWORD dwIP;
((CIPAddressCtrl*)GetDlgItem(IDC_IPADDRESS1))->GetAddress(dwIP);

SOCKADDR_IN addrTo;
addrTo.sin_family = AF_INET;
addrTo.sin_port =htons(6000);//这里的问题,监听的是6000号端口,当然是向这个端口发数据
addrTo.sin_addr.S_un.S_addr = htonl(dwIP);
。。。。
}

sese53 2009-02-26
  • 打赏
  • 举报
回复
我本来就是停了的。我直接用控制台做个最简单的UDP程序,也是6000端口,也是成功了的。
有谁能帮助我一下么。可能是哪些问题?
瓶盒 2009-02-26
  • 打赏
  • 举报
回复
把XP自带的防火墙停了试一下。
sese53 2009-02-26
  • 打赏
  • 举报
回复
[Quote=引用 6 楼 vagrantisme 的回复:]
1、telnet 那个端口看是否在监听状态。根据你的现象一般是不在监听状态。
2、如果在监听状态,则加TRACE或单步跟踪,很简单就发现问题之所在了。


祝好运
[/Quote]
打开我的程序后,我用netstat -a发现6000端口是在被监听,
关闭后,没有被监听了
sese53 2009-02-26
  • 打赏
  • 举报
回复
我知道这个错误代码的意思,但是完全不能解决啊。我在第一个程序说了是在那个IF分支中退出的,能有好心人帮帮忙么
vagrantisme 2009-02-26
  • 打赏
  • 举报
回复
1、telnet 那个端口看是否在监听状态。根据你的现象一般是不在监听状态。
2、如果在监听状态,则加TRACE或单步跟踪,很简单就发现问题之所在了。


祝好运
oyljerry 2009-02-26
  • 打赏
  • 举报
回复
10054,表示"远程主机强迫关闭了现有的一个连接"
ip地址,端口等是否被防火墙等阻止
D0ckLng 2009-02-26
  • 打赏
  • 举报
回复
你的错误代码表示WSAECONNRESET 连接被重置.
能帮的就这点了.
龙凤呈祥焱 2009-02-26
  • 打赏
  • 举报
回复
眼花
sese53 2009-02-26
  • 打赏
  • 举报
回复
环境是XP,在VS05中完成的。
sese53 2009-02-26
  • 打赏
  • 举报
回复
对了。我的机器是xp,环境是VS2005,已经退出所有杀毒软件和360

18,356

社区成员

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

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